1 /* 2 * Copyright (c) 2005, 2025, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. 8 * 9 * This code is distributed in the hope that it will be useful, but WITHOUT 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 12 * version 2 for more details (a copy is included in the LICENSE file that 13 * accompanied this code). 14 * 15 * You should have received a copy of the GNU General Public License version 16 * 2 along with this work; if not, write to the Free Software Foundation, 17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 18 * 19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 20 * or visit www.oracle.com if you need additional information or have any 21 * questions. 22 */ 23 24 /* 25 * @test 26 * @bug 6207984 6272521 6192552 6269713 6197726 6260652 5073546 4137464 27 * 4155650 4216399 4294891 6282555 6318622 6355327 6383475 6420753 28 * 6431845 4802633 6570566 6570575 6570631 6570924 6691185 6691215 29 * 4802647 7123424 8024709 8193128 8327858 8346307 30 * @summary Run many tests on many Collection and Map implementations 31 * @author Martin Buchholz 32 * @modules java.base/java.util:open 33 * @enablePreview 34 * @run main MOAT 35 * @run main MOAT --enable-preview 36 * @key randomness 37 */ 38 39 /* Mother Of All (Collection) Tests 40 * 41 * Testing of collection classes is often spotty, because many tests 42 * need to be performed on many implementations, but the onus on 43 * writing the tests falls on the engineer introducing the new 44 * implementation. 45 * 46 * The idea of this mega-test is that: 47 * 48 * An engineer adding a new collection implementation could simply add 49 * their new implementation to a list of implementations in this 50 * test's main method. Any general purpose Collection<Integer> or 51 * Map<Integer,Integer> class is appropriate. 52 * 53 * An engineer fixing a regression could add their regression test here and 54 * simultaneously test all other implementations. 55 */ 56 57 import java.io.*; 58 import java.util.*; 59 import java.util.concurrent.*; 60 import static java.util.Collections.*; 61 import java.lang.reflect.*; 62 import java.util.stream.Collectors; 63 import java.util.stream.Stream; 64 65 public class MOAT { 66 // Collections under test must not be initialized to contain this value, 67 // and maps under test must not contain this value as a key. 68 // It's used as a sentinel for absent-element testing. 69 static final int ABSENT_VALUE = 778347983; 70 71 static final Integer[] integerArray; 72 static { 73 Integer[] ia = new Integer[20]; 74 // fill with 1..20 inclusive 75 for (int i = 0; i < ia.length; i++) { 76 ia[i] = i + 1; 77 } 78 integerArray = ia; 79 } 80 81 public static void realMain(String[] args) { 82 83 testCollection(new NewAbstractCollection<Integer>()); 84 testCollection(new NewAbstractSet<Integer>()); 85 testCollection(new LinkedHashSet<Integer>()); 86 testCollection(new HashSet<Integer>()); 87 testCollection(new Vector<Integer>()); 88 testCollection(new Vector<Integer>().subList(0,0)); 89 testCollection(new ArrayDeque<Integer>()); 90 testCollection(new ArrayList<Integer>()); 91 testCollection(new ArrayList<Integer>().subList(0,0)); 92 testCollection(new LinkedList<Integer>()); 93 testCollection(new LinkedList<Integer>().subList(0,0)); 94 testCollection(new TreeSet<Integer>()); 95 testCollection(Collections.checkedList(new ArrayList<Integer>(), Integer.class)); 96 testCollection(Collections.synchronizedList(new ArrayList<Integer>())); 97 testCollection(Collections.checkedSet(new HashSet<Integer>(), Integer.class)); 98 testCollection(Collections.checkedSortedSet(new TreeSet<Integer>(), Integer.class)); 99 testCollection(Collections.checkedNavigableSet(new TreeSet<Integer>(), Integer.class)); 100 testCollection(Collections.synchronizedSet(new HashSet<Integer>())); 101 testCollection(Collections.synchronizedSortedSet(new TreeSet<Integer>())); 102 testCollection(Collections.synchronizedNavigableSet(new TreeSet<Integer>())); 103 104 testCollection(new CopyOnWriteArrayList<Integer>()); 105 testCollection(new CopyOnWriteArrayList<Integer>().subList(0,0)); 106 testCollection(new CopyOnWriteArraySet<Integer>()); 107 testCollection(new PriorityQueue<Integer>()); 108 testCollection(new PriorityBlockingQueue<Integer>()); 109 testCollection(new ArrayBlockingQueue<Integer>(20)); 110 testCollection(new LinkedBlockingQueue<Integer>(20)); 111 testCollection(new LinkedBlockingDeque<Integer>(20)); 112 testCollection(new ConcurrentLinkedDeque<Integer>()); 113 testCollection(new ConcurrentLinkedQueue<Integer>()); 114 testCollection(new LinkedTransferQueue<Integer>()); 115 testCollection(new ConcurrentSkipListSet<Integer>()); 116 testCollection(Arrays.asList(new Integer(42))); 117 testCollection(Arrays.asList(1,2,3)); 118 testCollection(nCopies(25,1)); 119 testImmutableList(nCopies(25,1)); 120 121 testMap(new HashMap<Integer,Integer>()); 122 testMap(new LinkedHashMap<Integer,Integer>()); 123 124 // TODO: Add reliable support for WeakHashMap. 125 // This test is subject to very rare failures because the GC 126 // may remove unreferenced-keys from the map at any time. 127 // testMap(new WeakHashMap<Integer,Integer>()); 128 129 testMap(new IdentityHashMap<Integer,Integer>()); 130 testMap(new TreeMap<Integer,Integer>()); 131 testMap(new Hashtable<Integer,Integer>()); 132 testMap(new ConcurrentHashMap<Integer,Integer>(10, 0.5f)); 133 testMap(new ConcurrentSkipListMap<Integer,Integer>()); 134 testMap(Collections.checkedMap(new HashMap<Integer,Integer>(), Integer.class, Integer.class)); 135 testMap(Collections.checkedSortedMap(new TreeMap<Integer,Integer>(), Integer.class, Integer.class)); 136 testMap(Collections.checkedNavigableMap(new TreeMap<Integer,Integer>(), Integer.class, Integer.class)); 137 testMap(Collections.synchronizedMap(new HashMap<Integer,Integer>())); 138 testMap(Collections.synchronizedSortedMap(new TreeMap<Integer,Integer>())); 139 testMap(Collections.synchronizedNavigableMap(new TreeMap<Integer,Integer>())); 140 141 // Unmodifiable wrappers 142 testImmutableSet(unmodifiableSet(new HashSet<>(Arrays.asList(1,2,3))), 99); 143 testImmutableSet(AccessFlag.maskToAccessFlags(0, AccessFlag.Location.CLASS), AccessFlag.ABSTRACT); 144 testImmutableSet(AccessFlag.maskToAccessFlags(Modifier.PUBLIC | Modifier.STATIC | Modifier.SYNCHRONIZED, AccessFlag.Location.METHOD), AccessFlag.ABSTRACT); 145 testImmutableList(unmodifiableList(Arrays.asList(1,2,3))); 146 testImmutableMap(unmodifiableMap(Collections.singletonMap(1,2))); 147 testImmutableSeqColl(unmodifiableSequencedCollection(Arrays.asList(1,2,3)), 99); 148 testImmutableSeqColl(unmodifiableSequencedSet(new LinkedHashSet<>(Arrays.asList(1,2,3))), 99); 149 var lhm = new LinkedHashMap<Integer,Integer>(); lhm.put(1,2); lhm.put(3, 4); 150 testImmutableSeqMap(unmodifiableSequencedMap(lhm)); 151 testCollMutatorsAlwaysThrow(unmodifiableSet(new HashSet<>(Arrays.asList(1,2,3)))); 152 testCollMutatorsAlwaysThrow(unmodifiableSet(Collections.emptySet())); 153 testEmptyCollMutatorsAlwaysThrow(unmodifiableSet(Collections.emptySet())); 154 testListMutatorsAlwaysThrow(unmodifiableList(Arrays.asList(1,2,3))); 155 testListMutatorsAlwaysThrow(unmodifiableList(Collections.emptyList())); 156 testEmptyListMutatorsAlwaysThrow(unmodifiableList(Collections.emptyList())); 157 testMapMutatorsAlwaysThrow(unmodifiableMap(Collections.singletonMap(1,2))); 158 testMapMutatorsAlwaysThrow(unmodifiableMap(Collections.emptyMap())); 159 testEmptyMapMutatorsAlwaysThrow(unmodifiableMap(Collections.emptyMap())); 160 161 // Empty collections 162 final List<Integer> emptyArray = Arrays.asList(new Integer[]{}); 163 testCollection(emptyArray); 164 testEmptyList(emptyArray); 165 THROWS(IndexOutOfBoundsException.class, () -> emptyArray.set(0,1)); 166 THROWS(UnsupportedOperationException.class, () -> emptyArray.add(0,1)); 167 168 List<Integer> noOne = nCopies(0,1); 169 testCollection(noOne); 170 testEmptyList(noOne); 171 testImmutableList(noOne); 172 173 Set<Integer> emptySet = emptySet(); 174 testCollection(emptySet); 175 testEmptySet(emptySet); 176 testEmptySet(EMPTY_SET); 177 testEmptySet(Collections.emptySet()); 178 testEmptySet(Collections.emptySortedSet()); 179 testEmptySet(Collections.emptyNavigableSet()); 180 testImmutableSet(emptySet, 99); 181 182 List<Integer> emptyList = emptyList(); 183 testCollection(emptyList); 184 testEmptyList(emptyList); 185 testEmptyList(EMPTY_LIST); 186 testEmptyList(Collections.emptyList()); 187 testImmutableList(emptyList); 188 189 Map<Integer,Integer> emptyMap = emptyMap(); 190 testMap(emptyMap); 191 testEmptyMap(emptyMap); 192 testEmptyMap(EMPTY_MAP); 193 testEmptyMap(Collections.emptyMap()); 194 testEmptyMap(Collections.emptySortedMap()); 195 testEmptyMap(Collections.emptyNavigableMap()); 196 testImmutableMap(emptyMap); 197 testImmutableMap(Collections.emptyMap()); 198 testImmutableMap(Collections.emptySortedMap()); 199 testImmutableMap(Collections.emptyNavigableMap()); 200 201 // Singleton collections 202 Set<Integer> singletonSet = singleton(1); 203 equal(singletonSet.size(), 1); 204 testCollection(singletonSet); 205 testImmutableSet(singletonSet, 99); 206 207 List<Integer> singletonList = singletonList(1); 208 equal(singletonList.size(), 1); 209 testCollection(singletonList); 210 testImmutableList(singletonList); 211 testImmutableList(singletonList.subList(0,1)); 212 testImmutableList(singletonList.subList(0,1).subList(0,1)); 213 testEmptyList(singletonList.subList(0,0)); 214 testEmptyList(singletonList.subList(0,0).subList(0,0)); 215 216 Map<Integer,Integer> singletonMap = singletonMap(1,2); 217 equal(singletonMap.size(), 1); 218 testMap(singletonMap); 219 testImmutableMap(singletonMap); 220 221 // Immutable List 222 testEmptyList(List.of()); 223 testEmptyList(List.of().subList(0,0)); 224 testEmptyList(StableValue.list(0, i -> i)); 225 testEmptyList(StableValue.list(3, i -> i).subList(0, 0)); 226 testListMutatorsAlwaysThrow(List.of()); 227 testListMutatorsAlwaysThrow(List.<Integer>of().subList(0,0)); 228 testListMutatorsAlwaysThrow(StableValue.list(0, i -> i)); 229 testEmptyListMutatorsAlwaysThrow(List.of()); 230 testEmptyListMutatorsAlwaysThrow(List.<Integer>of().subList(0,0)); 231 testEmptyListMutatorsAlwaysThrow(StableValue.list(0, i -> i)); 232 testEmptyListMutatorsAlwaysThrow(StableValue.list(3, i -> i).subList(0, 0)); 233 for (List<Integer> list : Arrays.asList( 234 List.<Integer>of(), 235 List.of(1), 236 List.of(1, 2), 237 List.of(1, 2, 3), 238 List.of(1, 2, 3, 4), 239 List.of(1, 2, 3, 4, 5), 240 List.of(1, 2, 3, 4, 5, 6), 241 List.of(1, 2, 3, 4, 5, 6, 7), 242 List.of(1, 2, 3, 4, 5, 6, 7, 8), 243 List.of(1, 2, 3, 4, 5, 6, 7, 8, 9), 244 List.of(1, 2, 3, 4, 5, 6, 7, 8, 9, 10), 245 List.of(integerArray), 246 Stream.<Integer>empty().toList(), 247 Stream.of(1).toList(), 248 Stream.of(1, 2).toList(), 249 Stream.of(1, 2, 3).toList(), 250 Stream.of(1, 2, 3, 4).toList(), 251 Stream.of((Integer)null).toList(), 252 Stream.of(1, null).toList(), 253 Stream.of(1, null, 3).toList(), 254 Stream.of(1, null, 3, 4).toList(), 255 StableValue.list(0, i -> i), 256 StableValue.list(3, i -> i), 257 StableValue.list(10, i -> i))) { 258 testCollection(list); 259 testImmutableList(list); 260 testListMutatorsAlwaysThrow(list); 261 testImmutableListMutatorsAlwaysThrow(list); 262 if (list.size() >= 1) { 263 // test subLists 264 List<Integer> headList = list.subList(0, list.size() - 1); 265 List<Integer> tailList = list.subList(1, list.size()); 266 testCollection(headList); 267 testCollection(tailList); 268 testImmutableList(headList); 269 testImmutableList(tailList); 270 testListMutatorsAlwaysThrow(headList); 271 testListMutatorsAlwaysThrow(tailList); 272 } 273 } 274 275 List<Integer> listCopy = List.copyOf(Arrays.asList(1, 2, 3)); 276 testCollection(listCopy); 277 testImmutableList(listCopy); 278 testListMutatorsAlwaysThrow(listCopy); 279 280 List<Integer> listCollected = Stream.of(1, 2, 3).collect(Collectors.toUnmodifiableList()); 281 equal(listCollected, List.of(1, 2, 3)); 282 testCollection(listCollected); 283 testImmutableList(listCollected); 284 testListMutatorsAlwaysThrow(listCollected); 285 286 // List indexOf / lastIndexOf 287 288 // 0 element 289 System.out.println("testListIndexOf size 0"); 290 testListIndexOf(-1, -1); 291 292 System.out.println("testListIndexOf size 1"); 293 testListIndexOf(-1, -1, 0); 294 testListIndexOf(0, 0, 1); 295 296 System.out.println("testListIndexOf size 2"); 297 testListIndexOf(-1, -1, 0, 0); 298 testListIndexOf(0, 0, 1, 0); 299 testListIndexOf(0, 1, 1, 1); 300 testListIndexOf(1, 1, 0, 1); 301 302 303 System.out.println("testListIndexOf size 3"); 304 testListIndexOf(-1, -1, 0, 0, 0); 305 testListIndexOf(0, 0, 1, 0, 0); 306 testListIndexOf(0, 1, 1, 1, 0); 307 testListIndexOf(1, 2, 0, 1, 1); 308 testListIndexOf(2, 2, 0, 0, 1); 309 310 System.out.println("testListIndexOf size N"); 311 testListIndexOf(-1, -1, 0, 0, 0, 0, 0, 0, 0); 312 testListIndexOf(2, 6, 0, 0, 1, 0, 1, 0, 1); 313 testListIndexOf(4, 4, 0, 0, 0, 0, 1, 0, 0); 314 testListIndexOf(0, 6, 1, 1, 1, 1, 1, 1, 1); 315 testListIndexOf(0, 7, 1, 1, 1, 1, 1, 1, 1, 1); 316 testListIndexOf(0, 8, 1, 1, 1, 1, 1, 1, 1, 1, 1); 317 testListIndexOf(0, 9, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1); 318 testListIndexOf(0, 10, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1); 319 testListIndexOf(0, 11, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1); 320 testListIndexOf(0, 12, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1); 321 testListIndexOf(12, 12, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1); 322 testListIndexOf(-1, -1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0); 323 324 // Immutable Set 325 testEmptySet(Set.of()); 326 testCollMutatorsAlwaysThrow(Set.of()); 327 testEmptyCollMutatorsAlwaysThrow(Set.of()); 328 for (Set<Integer> set : Arrays.asList( 329 Set.<Integer>of(), 330 Set.of(1), 331 Set.of(1, 2), 332 Set.of(1, 2, 3), 333 Set.of(1, 2, 3, 4), 334 Set.of(1, 2, 3, 4, 5), 335 Set.of(1, 2, 3, 4, 5, 6), 336 Set.of(1, 2, 3, 4, 5, 6, 7), 337 Set.of(1, 2, 3, 4, 5, 6, 7, 8), 338 Set.of(1, 2, 3, 4, 5, 6, 7, 8, 9), 339 Set.of(1, 2, 3, 4, 5, 6, 7, 8, 9, 10), 340 Set.of(integerArray))) { 341 testCollection(set); 342 testImmutableSet(set, 99); 343 testCollMutatorsAlwaysThrow(set); 344 } 345 346 Set<Integer> setCopy = Set.copyOf(Arrays.asList(1, 2, 3)); 347 testCollection(setCopy); 348 testImmutableSet(setCopy, 99); 349 testCollMutatorsAlwaysThrow(setCopy); 350 351 Set<Integer> setCollected = Stream.of(1, 1, 2, 3, 2, 3) 352 .collect(Collectors.toUnmodifiableSet()); 353 equal(setCollected, Set.of(1, 2, 3)); 354 testCollection(setCollected); 355 testImmutableSet(setCollected, 99); 356 testCollMutatorsAlwaysThrow(setCollected); 357 358 // Immutable Map 359 360 @SuppressWarnings("unchecked") 361 Map.Entry<Integer,Integer>[] ea = (Map.Entry<Integer,Integer>[])new Map.Entry<?,?>[20]; 362 for (int i = 0; i < ea.length; i++) { 363 ea[i] = Map.entry(i+1, i+101); 364 } 365 366 testEmptyMap(Map.of()); 367 testMapMutatorsAlwaysThrow(Map.of()); 368 testEmptyMapMutatorsAlwaysThrow(Map.of()); 369 testEmptyMap(StableValue.map(Set.of(), k -> k)); 370 testMapMutatorsAlwaysThrow(StableValue.map(Set.of(), k -> k)); 371 testEmptyMapMutatorsAlwaysThrow(StableValue.map(Set.of(), k -> k)); 372 for (Map<Integer,Integer> map : Arrays.asList( 373 Map.<Integer,Integer>of(), 374 Map.of(1, 101), 375 Map.of(1, 101, 2, 202), 376 Map.of(1, 101, 2, 202, 3, 303), 377 Map.of(1, 101, 2, 202, 3, 303, 4, 404), 378 Map.of(1, 101, 2, 202, 3, 303, 4, 404, 5, 505), 379 Map.of(1, 101, 2, 202, 3, 303, 4, 404, 5, 505, 6, 606), 380 Map.of(1, 101, 2, 202, 3, 303, 4, 404, 5, 505, 6, 606, 7, 707), 381 Map.of(1, 101, 2, 202, 3, 303, 4, 404, 5, 505, 6, 606, 7, 707, 8, 808), 382 Map.of(1, 101, 2, 202, 3, 303, 4, 404, 5, 505, 6, 606, 7, 707, 8, 808, 9, 909), 383 Map.of(1, 101, 2, 202, 3, 303, 4, 404, 5, 505, 6, 606, 7, 707, 8, 808, 9, 909, 10, 1010), 384 Map.ofEntries(ea), 385 StableValue.map(Set.<Integer>of(), k -> k), 386 StableValue.map(Set.of(1), k -> k), 387 StableValue.map(Set.of(1, 2, 3), k -> k))) { 388 testMap(map); 389 testImmutableMap(map); 390 testMapMutatorsAlwaysThrow(map); 391 } 392 393 Map<Integer,Integer> mapCopy = Map.copyOf(new HashMap<>(Map.of(1, 101, 2, 202, 3, 303))); 394 testMap(mapCopy); 395 testImmutableMap(mapCopy); 396 testMapMutatorsAlwaysThrow(mapCopy); 397 398 Map<Integer,Integer> mapCollected1 = 399 Stream.of(1, 2, 3) 400 .collect(Collectors.toUnmodifiableMap(i -> i, i -> 101 * i)); 401 equal(mapCollected1, Map.of(1, 101, 2, 202, 3, 303)); 402 testMap(mapCollected1); 403 testImmutableMap(mapCollected1); 404 testMapMutatorsAlwaysThrow(mapCollected1); 405 406 try { 407 Stream.of(1, 1, 2, 3, 2, 3) 408 .collect(Collectors.toUnmodifiableMap(i -> i, i -> 101 * i)); 409 fail("duplicates should have thrown an exception"); 410 } catch (IllegalStateException ise) { 411 pass(); 412 } 413 414 Map<Integer,Integer> mapCollected2 = 415 Stream.of(1, 1, 2, 3, 2, 3) 416 .collect(Collectors.toUnmodifiableMap(i -> i, i -> 101 * i, Integer::sum)); 417 equal(mapCollected2, Map.of(1, 202, 2, 404, 3, 606)); 418 testMap(mapCollected2); 419 testImmutableMap(mapCollected2); 420 testMapMutatorsAlwaysThrow(mapCollected2); 421 } 422 423 private static void checkContainsSelf(Collection<Integer> c) { 424 check(c.containsAll(c)); 425 check(c.containsAll(Arrays.asList(c.toArray()))); 426 check(c.containsAll(Arrays.asList(c.toArray(new Integer[0])))); 427 } 428 429 private static void checkContainsEmpty(Collection<Integer> c) { 430 check(c.containsAll(new ArrayList<Integer>())); 431 } 432 433 private static void checkUnique(Set<Integer> s) { 434 for (Integer i : s) { 435 int count = 0; 436 for (Integer j : s) { 437 if (Objects.equals(i,j)) 438 ++count; 439 } 440 check(count == 1); 441 } 442 } 443 444 private static <T> void testEmptyCollection(Collection<T> c) { 445 check(c.isEmpty()); 446 equal(c.size(), 0); 447 equal(c.toString(),"[]"); 448 equal(c.toArray().length, 0); 449 equal(c.toArray(new Object[0]).length, 0); 450 equal(c.toArray(Object[]::new).length, 0); 451 check(c.toArray(new Object[]{42})[0] == null); 452 453 Object[] a = new Object[1]; a[0] = Boolean.TRUE; 454 equal(c.toArray(a), a); 455 equal(a[0], null); 456 testEmptyIterator(c.iterator()); 457 } 458 459 static <T> void testEmptyIterator(final Iterator<T> it) { 460 if (rnd.nextBoolean()) 461 check(! it.hasNext()); 462 463 THROWS(NoSuchElementException.class, () -> it.next()); 464 465 try { it.remove(); } 466 catch (IllegalStateException ignored) { pass(); } 467 catch (UnsupportedOperationException ignored) { pass(); } 468 catch (Throwable t) { unexpected(t); } 469 470 if (rnd.nextBoolean()) 471 check(! it.hasNext()); 472 } 473 474 private static void testEmptyList(List<?> c) { 475 testEmptyCollection(c); 476 equal(c.hashCode(), 1); 477 equal2(c, Collections.<Integer>emptyList()); 478 } 479 480 private static <T> void testEmptySet(Set<T> c) { 481 testEmptyCollection(c); 482 equal(c.hashCode(), 0); 483 equal2(c, Collections.<Integer>emptySet()); 484 if (c instanceof NavigableSet<?>) 485 testEmptyIterator(((NavigableSet<T>)c).descendingIterator()); 486 } 487 488 private static <T> void testImmutableCollection(final Collection<T> c, T t) { 489 THROWS(UnsupportedOperationException.class, 490 () -> c.add(t), 491 () -> c.addAll(singleton(t))); 492 if (! c.isEmpty()) { 493 final T first = c.iterator().next(); 494 THROWS(UnsupportedOperationException.class, 495 () -> c.clear(), 496 () -> c.remove(first), 497 () -> c.removeAll(singleton(first)), 498 () -> c.retainAll(emptyList())); 499 } else { 500 testEmptyIterator(c.iterator()); 501 } 502 testForEachMatch(c); 503 testSpliteratorMatch(c); 504 } 505 506 private static <T> void testImmutableSeqColl(final SequencedCollection<T> c, T t) { 507 SequencedCollection<T> r = c.reversed(); 508 testImmutableCollection(c, t); 509 testImmutableCollection(r, t); 510 THROWS(UnsupportedOperationException.class, 511 () -> c.addFirst(t), 512 () -> c.addLast(t), 513 () -> r.addFirst(t), 514 () -> r.addLast(t)); 515 if (! c.isEmpty()) { 516 THROWS(UnsupportedOperationException.class, 517 () -> c.removeFirst(), 518 () -> c.removeLast(), 519 () -> r.removeFirst(), 520 () -> r.removeLast()); 521 } 522 } 523 524 private static <T> void testImmutableSet(final Set<T> c, T t) { 525 testImmutableCollection(c, t); 526 } 527 528 private static <T> void testImmutableSeqSet(final SequencedSet<T> c, T t) { 529 testImmutableSeqColl(c, t); 530 } 531 532 private static void testImmutableList(final List<Integer> c) { 533 testList(c); 534 testImmutableCollection(c, 42); 535 testImmutableSeqColl(c, 42); 536 THROWS(UnsupportedOperationException.class, 537 () -> c.set(0,42), 538 () -> c.add(0,42), 539 () -> c.addAll(0,singleton(86))); 540 if (! c.isEmpty()) 541 THROWS(UnsupportedOperationException.class, 542 () -> { Iterator<Integer> it = c.iterator(); 543 it.next(); 544 it.remove(); }, 545 () -> { ListIterator<Integer> it = c.listIterator(); 546 it.next(); 547 it.remove(); }); 548 } 549 550 /** 551 * Test that calling a mutator always throws UOE, even if the mutator 552 * wouldn't actually do anything, given its arguments. 553 * 554 * @param c the collection instance to test 555 */ 556 private static void testCollMutatorsAlwaysThrow(Collection<Integer> c) { 557 THROWS(UnsupportedOperationException.class, 558 () -> c.addAll(Collections.emptyList()), 559 () -> c.remove(ABSENT_VALUE), 560 () -> c.removeAll(Collections.emptyList()), 561 () -> c.removeIf(x -> false), 562 () -> c.retainAll(c)); 563 } 564 565 // Ensures forEach supplies in the same order as the iterator 566 private static <T> void testForEachMatch(Collection<T> c) { 567 var itr = c.iterator(); 568 int[] index = {0}; 569 c.forEach(item -> { 570 T itrNext = null; 571 if (!itr.hasNext() || !Objects.equals(itrNext = itr.next(), item)) { 572 fail("forEach and iterator mismatch at " + index[0] + " forEach: " + item + ", itr: " + itrNext); 573 } 574 index[0]++; 575 }); 576 if (itr.hasNext()) { 577 fail("forEach and iterator mismatch at tail, extras in itr"); 578 } 579 } 580 581 // Ensures spliterator returns in the same order as the iterator 582 private static <T> void testSpliteratorMatch(Collection<T> c) { 583 var itr = c.iterator(); 584 var split = c.spliterator(); 585 int[] index = {0}; 586 split.forEachRemaining(item -> { 587 T itrNext = null; 588 if (!itr.hasNext() || !Objects.equals(itrNext = itr.next(), item)) { 589 fail("iterator and spliterator mismatch at " + index[0] + " spliterator: " + item + ", itr: " + itrNext); 590 } 591 index[0]++; 592 }); 593 if (itr.hasNext()) { 594 fail("iterator and spliterator mismatch at tail, extra item in itr"); 595 } 596 } 597 598 /** 599 * Test that calling a mutator always throws UOE, even if the mutator 600 * wouldn't actually do anything on an empty collection. 601 * 602 * @param c the collection instance to test, must be empty 603 */ 604 private static void testEmptyCollMutatorsAlwaysThrow(Collection<Integer> c) { 605 if (! c.isEmpty()) { 606 fail("collection is not empty"); 607 } 608 THROWS(UnsupportedOperationException.class, 609 () -> c.clear()); 610 } 611 612 /** 613 * As above, for a list. 614 * 615 * @param c the list instance to test 616 */ 617 private static void testListMutatorsAlwaysThrow(List<Integer> c) { 618 testCollMutatorsAlwaysThrow(c); 619 THROWS(UnsupportedOperationException.class, 620 () -> c.addAll(0, Collections.emptyList())); 621 } 622 623 private static void testImmutableListMutatorsAlwaysThrow(List<Integer> c) { 624 THROWS(UnsupportedOperationException.class, 625 c::removeFirst, 626 c::removeLast); 627 } 628 629 /** 630 * As above, for an empty list. 631 * 632 * @param c the list instance to test, must be empty 633 */ 634 private static void testEmptyListMutatorsAlwaysThrow(List<Integer> c) { 635 if (! c.isEmpty()) { 636 fail("list is not empty"); 637 } 638 testEmptyCollMutatorsAlwaysThrow(c); 639 THROWS(UnsupportedOperationException.class, 640 () -> c.replaceAll(x -> x), 641 () -> c.sort(null)); 642 } 643 644 /** 645 * As above, for a map. 646 * 647 * @param m the map instance to test 648 */ 649 private static void testMapMutatorsAlwaysThrow(Map<Integer,Integer> m) { 650 THROWS(UnsupportedOperationException.class, 651 () -> m.compute(ABSENT_VALUE, (k, v) -> null), 652 () -> m.computeIfAbsent(ABSENT_VALUE, k -> null), 653 () -> m.computeIfPresent(ABSENT_VALUE, (k, v) -> null), 654 () -> m.merge(ABSENT_VALUE, 0, (k, v) -> null), 655 () -> m.putAll(Collections.emptyMap()), 656 () -> m.remove(ABSENT_VALUE), 657 () -> m.remove(ABSENT_VALUE, 0), 658 () -> m.replace(ABSENT_VALUE, 0), 659 () -> m.replace(ABSENT_VALUE, 0, 1)); 660 } 661 662 /** 663 * As above, for an empty map. 664 * 665 * @param map the map instance to test, must be empty 666 */ 667 private static void testEmptyMapMutatorsAlwaysThrow(Map<Integer,Integer> m) { 668 if (! m.isEmpty()) { 669 fail("map is not empty"); 670 } 671 THROWS(UnsupportedOperationException.class, 672 () -> m.clear(), 673 () -> m.replaceAll((k, v) -> v)); 674 } 675 676 private static void clear(Collection<Integer> c) { 677 try { c.clear(); } 678 catch (Throwable t) { unexpected(t); } 679 testEmptyCollection(c); 680 } 681 682 private static <K,V> void testEmptyMap(final Map<K,V> m) { 683 check(m.isEmpty()); 684 equal(m.size(), 0); 685 equal(m.toString(),"{}"); 686 testEmptySet(m.keySet()); 687 testEmptySet(m.entrySet()); 688 testEmptyCollection(m.values()); 689 690 try { check(! m.containsValue(null)); } 691 catch (NullPointerException ignored) { /* OK */ } 692 try { check(! m.containsKey(null)); } 693 catch (NullPointerException ignored) { /* OK */ } 694 check(! m.containsValue(1)); 695 check(! m.containsKey(1)); 696 } 697 698 private static void testImmutableMapEntry(final Map.Entry<Integer,Integer> me) { 699 Integer key = me.getKey(); 700 Integer val = me.getValue(); 701 THROWS(UnsupportedOperationException.class, 702 () -> me.setValue(3)); 703 equal(key, me.getKey()); 704 equal(val, me.getValue()); 705 } 706 707 private static void testImmutableMap(final Map<Integer,Integer> m) { 708 THROWS(UnsupportedOperationException.class, 709 () -> m.put(1,1), 710 () -> m.putAll(singletonMap(1,1))); 711 if (! m.isEmpty()) { 712 final Integer first = m.keySet().iterator().next(); 713 THROWS(UnsupportedOperationException.class, 714 () -> m.remove(first), 715 () -> m.clear()); 716 testImmutableMapEntry(m.entrySet().iterator().next()); 717 } 718 testImmutableSet(m.keySet(), 99); 719 testImmutableCollection(m.values(), 99); 720 testImmutableSet(m.entrySet(), Map.entry(42, 43)); 721 } 722 723 private static void testImmutableSeqMap(final SequencedMap<Integer,Integer> m) { 724 SequencedMap<Integer,Integer> r = m.reversed(); 725 testImmutableMap(m); 726 testImmutableMap(r); 727 THROWS(UnsupportedOperationException.class, 728 () -> m.putFirst(0, 0), 729 () -> m.putLast(0, 0), 730 () -> r.putFirst(0, 0), 731 () -> r.putLast(0, 0)); 732 if (! m.isEmpty()) { 733 THROWS(UnsupportedOperationException.class, 734 () -> m.pollFirstEntry(), 735 () -> m.pollLastEntry(), 736 () -> r.pollFirstEntry(), 737 () -> r.pollLastEntry()); 738 testImmutableMapEntry(m.sequencedEntrySet().getFirst()); 739 testImmutableMapEntry(r.sequencedEntrySet().getFirst()); 740 testImmutableMapEntry(m.sequencedEntrySet().reversed().getFirst()); 741 testImmutableMapEntry(r.sequencedEntrySet().reversed().getFirst()); 742 } 743 testImmutableSeqSet(m.sequencedKeySet(), 99); 744 testImmutableSeqColl(m.sequencedValues(), 99); 745 testImmutableSeqSet(m.sequencedEntrySet(), Map.entry(42, 43)); 746 testImmutableSeqSet(r.sequencedKeySet(), 99); 747 testImmutableSeqColl(r.sequencedValues(), 99); 748 testImmutableSeqSet(r.sequencedEntrySet(), Map.entry(42, 43)); 749 } 750 751 private static void clear(Map<?,?> m) { 752 try { m.clear(); } 753 catch (Throwable t) { unexpected(t); } 754 testEmptyMap(m); 755 } 756 757 private static void oneElement(Collection<Integer> c) { 758 clear(c); 759 try { 760 check(c.add(-42)); 761 equal(c.toString(), "[-42]"); 762 if (c instanceof Set) check(! c.add(-42)); 763 } catch (Throwable t) { unexpected(t); } 764 check(! c.isEmpty()); check(c.size() == 1); 765 } 766 767 private static boolean supportsAdd(Collection<Integer> c) { 768 try { check(c.add(ABSENT_VALUE)); } 769 catch (UnsupportedOperationException t) { return false; } 770 catch (Throwable t) { unexpected(t); } 771 772 try { 773 check(c.contains(ABSENT_VALUE)); 774 check(c.remove(ABSENT_VALUE)); 775 } catch (Throwable t) { unexpected(t); } 776 return true; 777 } 778 779 private static boolean supportsRemove(Collection<Integer> c) { 780 try { check(! c.remove(ABSENT_VALUE)); } 781 catch (UnsupportedOperationException t) { return false; } 782 catch (Throwable t) { unexpected(t); } 783 return true; 784 } 785 786 // 6260652: (coll) Arrays.asList(x).toArray().getClass() 787 // should be Object[].class 788 // Fixed in jdk9, but not jdk8 ... 789 static final boolean needToWorkAround6260652 = 790 Arrays.asList("").toArray().getClass() != Object[].class; 791 792 private static void checkFunctionalInvariants(Collection<Integer> c) { 793 try { 794 checkContainsSelf(c); 795 checkContainsEmpty(c); 796 check(c.size() != 0 ^ c.isEmpty()); 797 check(! c.contains(ABSENT_VALUE)); 798 799 { 800 int size = 0; 801 for (Integer i : c) size++; 802 check(c.size() == size); 803 } 804 805 if (c instanceof Set) { 806 checkUnique((Set<Integer>)c); 807 } 808 809 check(c.toArray().length == c.size()); 810 check(c.toArray().getClass() == Object[].class 811 || 812 (needToWorkAround6260652 && 813 c.getClass().getName().equals("java.util.Arrays$ArrayList"))); 814 for (int size : new int[]{0,1,c.size(), c.size()+1}) { 815 Integer[] a = c.toArray(new Integer[size]); 816 check((size > c.size()) || a.length == c.size()); 817 int i = 0; for (Integer j : c) check(a[i++] == j); 818 check((size <= c.size()) || (a[c.size()] == null)); 819 check(a.getClass() == Integer[].class); 820 } 821 822 { 823 Integer[] a = c.toArray(Integer[]::new); 824 equal(c.size(), a.length); 825 check(a.getClass() == Integer[].class); 826 check(Arrays.equals(c.toArray(new Integer[0]), a)); 827 } 828 829 check(c.equals(c)); 830 if (c instanceof Serializable) { 831 //System.out.printf("Serializing %s%n", c.getClass().getName()); 832 try { 833 Object clone = serialClone(c); 834 equal(c instanceof Serializable, 835 clone instanceof Serializable); 836 equal(c instanceof RandomAccess, 837 clone instanceof RandomAccess); 838 if ((c instanceof List) || (c instanceof Set)) 839 equal(c, clone); 840 else 841 equal(new HashSet<Integer>(c), 842 new HashSet<Integer>(serialClone(c))); 843 } catch (Error xxx) { 844 if (! (xxx.getCause() instanceof NotSerializableException)) 845 throw xxx; 846 } 847 } 848 } 849 catch (Throwable t) { unexpected(t); } 850 } 851 852 //---------------------------------------------------------------- 853 // If add(null) succeeds, contains(null) & remove(null) should succeed 854 //---------------------------------------------------------------- 855 private static void testNullElement(Collection<Integer> c) { 856 857 try { 858 check(c.add(null)); 859 if (c.size() == 1) 860 equal(c.toString(), "[null]"); 861 try { 862 checkFunctionalInvariants(c); 863 check(c.contains(null)); 864 check(c.remove(null)); 865 } 866 catch (Throwable t) { unexpected(t); } 867 } 868 catch (NullPointerException e) { /* OK */ } 869 catch (Throwable t) { unexpected(t); } 870 } 871 872 //---------------------------------------------------------------- 873 // If add("x") succeeds, contains("x") & remove("x") should succeed 874 //---------------------------------------------------------------- 875 @SuppressWarnings("unchecked") 876 private static void testStringElement(Collection<Integer> c) { 877 Collection x = (Collection)c; // Make type-unsafe 878 try { 879 check(x.add("x")); 880 try { 881 check(x.contains("x")); 882 check(x.remove("x")); 883 } catch (Throwable t) { unexpected(t); } 884 } 885 catch (ClassCastException e) { /* OK */ } 886 catch (Throwable t) { unexpected(t); } 887 } 888 889 private static void testConcurrentCollection(Collection<Integer> c) { 890 try { 891 c.add(1); 892 Iterator<Integer> it = c.iterator(); 893 check(it.hasNext()); 894 clear(c); 895 check(it.next() instanceof Integer); // No CME 896 check(c.isEmpty()); 897 } 898 catch (Throwable t) { unexpected(t); } 899 } 900 901 private static void testQueue(Queue<Integer> q) { 902 q.clear(); 903 for (int i = 0; i < 5; i++) { 904 testQueueAddRemove(q, null); 905 testQueueAddRemove(q, 537); 906 q.add(i); 907 } 908 equal(q.size(), 5); 909 checkFunctionalInvariants(q); 910 q.poll(); 911 equal(q.size(), 4); 912 checkFunctionalInvariants(q); 913 if ((q instanceof LinkedBlockingQueue) || 914 (q instanceof LinkedBlockingDeque) || 915 (q instanceof ConcurrentLinkedDeque) || 916 (q instanceof ConcurrentLinkedQueue)) { 917 testQueueIteratorRemove(q); 918 } 919 } 920 921 private static void testQueueAddRemove(final Queue<Integer> q, 922 final Integer e) { 923 final List<Integer> originalContents = new ArrayList<>(q); 924 final boolean isEmpty = q.isEmpty(); 925 final boolean isList = (q instanceof List); 926 final List asList = isList ? (List) q : null; 927 check(!q.contains(e)); 928 try { 929 q.add(e); 930 } catch (NullPointerException npe) { 931 check(e == null); 932 return; // Null elements not supported 933 } 934 check(q.contains(e)); 935 check(q.remove(e)); 936 check(!q.contains(e)); 937 equal(new ArrayList<Integer>(q), originalContents); 938 939 if (q instanceof Deque<?>) { 940 final Deque<Integer> deq = (Deque<Integer>) q; 941 final List<Integer> singleton = Collections.singletonList(e); 942 943 // insert, query, remove element at head 944 if (isEmpty) { 945 THROWS(NoSuchElementException.class, 946 () -> deq.getFirst(), 947 () -> deq.element(), 948 () -> deq.iterator().next()); 949 check(deq.peekFirst() == null); 950 check(deq.peek() == null); 951 } else { 952 check(deq.getFirst() != e); 953 check(deq.element() != e); 954 check(deq.iterator().next() != e); 955 check(deq.peekFirst() != e); 956 check(deq.peek() != e); 957 } 958 check(!deq.contains(e)); 959 check(!deq.removeFirstOccurrence(e)); 960 check(!deq.removeLastOccurrence(e)); 961 if (isList) { 962 check(asList.indexOf(e) == -1); 963 check(asList.lastIndexOf(e) == -1); 964 } 965 switch (rnd.nextInt(isList ? 4 : 3)) { 966 case 0: deq.addFirst(e); break; 967 case 1: check(deq.offerFirst(e)); break; 968 case 2: deq.push(e); break; 969 case 3: asList.add(0, e); break; 970 default: throw new AssertionError(); 971 } 972 check(deq.peekFirst() == e); 973 check(deq.getFirst() == e); 974 check(deq.element() == e); 975 check(deq.peek() == e); 976 check(deq.iterator().next() == e); 977 check(deq.contains(e)); 978 if (isList) { 979 check(asList.get(0) == e); 980 check(asList.indexOf(e) == 0); 981 check(asList.lastIndexOf(e) == 0); 982 check(asList.subList(0, 1).equals(singleton)); 983 } 984 switch (rnd.nextInt(isList ? 11 : 9)) { 985 case 0: check(deq.pollFirst() == e); break; 986 case 1: check(deq.removeFirst() == e); break; 987 case 2: check(deq.remove() == e); break; 988 case 3: check(deq.pop() == e); break; 989 case 4: check(deq.removeFirstOccurrence(e)); break; 990 case 5: check(deq.removeLastOccurrence(e)); break; 991 case 6: check(deq.remove(e)); break; 992 case 7: check(deq.removeAll(singleton)); break; 993 case 8: Iterator it = deq.iterator(); it.next(); it.remove(); break; 994 case 9: asList.remove(0); break; 995 case 10: asList.subList(0, 1).clear(); break; 996 default: throw new AssertionError(); 997 } 998 if (isEmpty) { 999 THROWS(NoSuchElementException.class, 1000 () -> deq.getFirst(), 1001 () -> deq.element(), 1002 () -> deq.iterator().next()); 1003 check(deq.peekFirst() == null); 1004 check(deq.peek() == null); 1005 } else { 1006 check(deq.getFirst() != e); 1007 check(deq.element() != e); 1008 check(deq.iterator().next() != e); 1009 check(deq.peekFirst() != e); 1010 check(deq.peek() != e); 1011 } 1012 check(!deq.contains(e)); 1013 check(!deq.removeFirstOccurrence(e)); 1014 check(!deq.removeLastOccurrence(e)); 1015 if (isList) { 1016 check(isEmpty || asList.get(0) != e); 1017 check(asList.indexOf(e) == -1); 1018 check(asList.lastIndexOf(e) == -1); 1019 } 1020 equal(new ArrayList<Integer>(deq), originalContents); 1021 1022 // insert, query, remove element at tail 1023 if (isEmpty) { 1024 check(deq.peekLast() == null); 1025 THROWS(NoSuchElementException.class, () -> deq.getLast()); 1026 } else { 1027 check(deq.peekLast() != e); 1028 check(deq.getLast() != e); 1029 } 1030 switch (rnd.nextInt(isList ? 6 : 4)) { 1031 case 0: deq.addLast(e); break; 1032 case 1: check(deq.offerLast(e)); break; 1033 case 2: check(deq.add(e)); break; 1034 case 3: deq.addAll(singleton); break; 1035 case 4: asList.addAll(deq.size(), singleton); break; 1036 case 5: asList.add(deq.size(), e); break; 1037 default: throw new AssertionError(); 1038 } 1039 check(deq.peekLast() == e); 1040 check(deq.getLast() == e); 1041 check(deq.contains(e)); 1042 if (isList) { 1043 ListIterator it = asList.listIterator(asList.size()); 1044 check(it.previous() == e); 1045 check(asList.get(asList.size() - 1) == e); 1046 check(asList.indexOf(e) == asList.size() - 1); 1047 check(asList.lastIndexOf(e) == asList.size() - 1); 1048 int size = asList.size(); 1049 check(asList.subList(size - 1, size).equals(singleton)); 1050 } 1051 switch (rnd.nextInt(isList ? 8 : 6)) { 1052 case 0: check(deq.pollLast() == e); break; 1053 case 1: check(deq.removeLast() == e); break; 1054 case 2: check(deq.removeFirstOccurrence(e)); break; 1055 case 3: check(deq.removeLastOccurrence(e)); break; 1056 case 4: check(deq.remove(e)); break; 1057 case 5: check(deq.removeAll(singleton)); break; 1058 case 6: asList.remove(asList.size() - 1); break; 1059 case 7: 1060 ListIterator it = asList.listIterator(asList.size()); 1061 it.previous(); 1062 it.remove(); 1063 break; 1064 default: throw new AssertionError(); 1065 } 1066 if (isEmpty) { 1067 check(deq.peekLast() == null); 1068 THROWS(NoSuchElementException.class, () -> deq.getLast()); 1069 } else { 1070 check(deq.peekLast() != e); 1071 check(deq.getLast() != e); 1072 } 1073 check(!deq.contains(e)); 1074 equal(new ArrayList<Integer>(deq), originalContents); 1075 1076 // Test operations on empty deque 1077 switch (rnd.nextInt(isList ? 4 : 2)) { 1078 case 0: deq.clear(); break; 1079 case 1: 1080 Iterator it = deq.iterator(); 1081 while (it.hasNext()) { 1082 it.next(); 1083 it.remove(); 1084 } 1085 break; 1086 case 2: asList.subList(0, asList.size()).clear(); break; 1087 case 3: 1088 ListIterator lit = asList.listIterator(asList.size()); 1089 while (lit.hasPrevious()) { 1090 lit.previous(); 1091 lit.remove(); 1092 } 1093 break; 1094 default: throw new AssertionError(); 1095 } 1096 testEmptyCollection(deq); 1097 check(!deq.iterator().hasNext()); 1098 if (isList) { 1099 check(!asList.listIterator().hasPrevious()); 1100 THROWS(NoSuchElementException.class, 1101 () -> asList.listIterator().previous()); 1102 } 1103 THROWS(NoSuchElementException.class, 1104 () -> deq.iterator().next(), 1105 () -> deq.element(), 1106 () -> deq.getFirst(), 1107 () -> deq.getLast(), 1108 () -> deq.pop(), 1109 () -> deq.remove(), 1110 () -> deq.removeFirst(), 1111 () -> deq.removeLast()); 1112 1113 check(deq.poll() == null); 1114 check(deq.pollFirst() == null); 1115 check(deq.pollLast() == null); 1116 check(deq.peek() == null); 1117 check(deq.peekFirst() == null); 1118 check(deq.peekLast() == null); 1119 check(!deq.removeFirstOccurrence(e)); 1120 check(!deq.removeLastOccurrence(e)); 1121 1122 check(deq.addAll(originalContents) == !isEmpty); 1123 equal(new ArrayList<Integer>(deq), originalContents); 1124 check(!deq.addAll(Collections.<Integer>emptyList())); 1125 equal(new ArrayList<Integer>(deq), originalContents); 1126 } 1127 } 1128 1129 private static void testQueueIteratorRemove(Queue<Integer> q) { 1130 System.err.printf("testQueueIteratorRemove %s%n", 1131 q.getClass().getSimpleName()); 1132 q.clear(); 1133 for (int i = 0; i < 5; i++) 1134 q.add(i); 1135 Iterator<Integer> it = q.iterator(); 1136 check(it.hasNext()); 1137 for (int i = 3; i >= 0; i--) 1138 q.remove(i); 1139 equal(it.next(), 0); 1140 equal(it.next(), 4); 1141 1142 q.clear(); 1143 for (int i = 0; i < 5; i++) 1144 q.add(i); 1145 it = q.iterator(); 1146 equal(it.next(), 0); 1147 check(it.hasNext()); 1148 for (int i = 1; i < 4; i++) 1149 q.remove(i); 1150 equal(it.next(), 1); 1151 equal(it.next(), 4); 1152 } 1153 1154 // for any array of integer values, check that the result of lastIndexOf(1) 1155 // and indexOf(1) match assumptions for all types of List<Integer> we can 1156 // construct 1157 private static void testListIndexOf(final int index, 1158 final int lastIndex, 1159 final Integer ... values) { 1160 if (values.length == 0) { 1161 checkListIndexOf(emptyList(), index, lastIndex); 1162 } else if (values.length == 1) { 1163 checkListIndexOf(singletonList(values[0]), index, lastIndex); 1164 checkListIndexOf(nCopies(25, values[0]), index, lastIndex == 0 ? 24 : -1); 1165 } 1166 List<Integer> l = List.of(values); 1167 checkListIndexOf(l, index, lastIndex); 1168 checkListIndexOf(Arrays.asList(values), index, lastIndex); 1169 checkListIndexOf(new ArrayList(l), index, lastIndex); 1170 checkListIndexOf(new LinkedList(l), index, lastIndex); 1171 checkListIndexOf(new Vector(l), index, lastIndex); 1172 checkListIndexOf(new CopyOnWriteArrayList(l), index, lastIndex); 1173 } 1174 1175 private static void checkListIndexOf(final List<Integer> list, 1176 final int index, 1177 final int lastIndex) { 1178 String msg = list.getClass().toString(); 1179 equal(list.indexOf(1), index, msg); 1180 equal(list.lastIndexOf(1), lastIndex, msg); 1181 equal(list.subList(0, list.size()).indexOf(1), index, msg); 1182 equal(list.subList(0, list.size()).lastIndexOf(1), lastIndex, msg); 1183 } 1184 1185 private static void testList(final List<Integer> l) { 1186 //---------------------------------------------------------------- 1187 // 4802633: (coll) AbstractList.addAll(-1,emptyCollection) 1188 // doesn't throw IndexOutOfBoundsException 1189 //---------------------------------------------------------------- 1190 try { 1191 l.addAll(-1, Collections.<Integer>emptyList()); 1192 fail("Expected IndexOutOfBoundsException not thrown"); 1193 } 1194 catch (UnsupportedOperationException ignored) {/* OK */} 1195 catch (IndexOutOfBoundsException ignored) {/* OK */} 1196 catch (Throwable t) { unexpected(t); } 1197 1198 // equal(l instanceof Serializable, 1199 // l.subList(0,0) instanceof Serializable); 1200 if (l.subList(0,0) instanceof Serializable) 1201 check(l instanceof Serializable); 1202 1203 equal(l instanceof RandomAccess, 1204 l.subList(0,0) instanceof RandomAccess); 1205 1206 l.iterator(); 1207 l.listIterator(); 1208 l.listIterator(0); 1209 l.listIterator(l.size()); 1210 THROWS(IndexOutOfBoundsException.class, 1211 () -> l.listIterator(-1), 1212 () -> l.listIterator(l.size() + 1)); 1213 1214 if (l instanceof AbstractList) { 1215 try { 1216 int size = l.size(); 1217 AbstractList<Integer> abList = (AbstractList<Integer>) l; 1218 Method m = AbstractList.class.getDeclaredMethod("removeRange", new Class[] { int.class, int.class }); 1219 m.setAccessible(true); 1220 m.invoke(abList, new Object[] { 0, 0 }); 1221 m.invoke(abList, new Object[] { size, size }); 1222 equal(size, l.size()); 1223 } 1224 catch (UnsupportedOperationException ignored) {/* OK */} 1225 catch (Throwable t) { unexpected(t); } 1226 } 1227 1228 int hashCode = 1; 1229 for (Integer i : l) 1230 hashCode = 31 * hashCode + (i == null ? 0 : i.hashCode()); 1231 check(l.hashCode() == hashCode); 1232 1233 var t = new ArrayList<>(l); 1234 check(t.equals(l)); 1235 check(l.equals(t)); 1236 } 1237 1238 private static void testCollection(Collection<Integer> c) { 1239 try { testCollection1(c); } 1240 catch (Throwable t) { unexpected(t); } 1241 } 1242 1243 private static void testCollection1(Collection<Integer> c) { 1244 1245 System.out.println("\n==> " + c.getClass().getName()); 1246 1247 checkFunctionalInvariants(c); 1248 1249 if (! supportsAdd(c)) return; 1250 //System.out.println("add() supported"); 1251 1252 if (c instanceof NavigableSet) { 1253 System.out.println("NavigableSet tests..."); 1254 1255 NavigableSet<Integer> ns = (NavigableSet<Integer>)c; 1256 testNavigableSet(ns); 1257 testNavigableSet(ns.headSet(6, false)); 1258 testNavigableSet(ns.headSet(5, true)); 1259 testNavigableSet(ns.tailSet(0, false)); 1260 testNavigableSet(ns.tailSet(1, true)); 1261 testNavigableSet(ns.subSet(0, false, 5, true)); 1262 testNavigableSet(ns.subSet(1, true, 6, false)); 1263 } 1264 1265 if (c instanceof Queue) 1266 testQueue((Queue<Integer>)c); 1267 1268 if (c instanceof List) 1269 testList((List<Integer>)c); 1270 1271 if (c instanceof Set) { 1272 int hashCode = 0; 1273 for (Integer i : c) 1274 hashCode = hashCode + (i == null ? 0 : i.hashCode()); 1275 check(c.hashCode() == hashCode); 1276 } 1277 1278 check(supportsRemove(c)); 1279 1280 try { 1281 oneElement(c); 1282 checkFunctionalInvariants(c); 1283 } 1284 catch (Throwable t) { unexpected(t); } 1285 1286 clear(c); testNullElement(c); 1287 oneElement(c); testNullElement(c); 1288 1289 clear(c); testStringElement(c); 1290 oneElement(c); testStringElement(c); 1291 1292 if (c.getClass().getName().matches(".*concurrent.*")) 1293 testConcurrentCollection(c); 1294 1295 //---------------------------------------------------------------- 1296 // The "all" operations should throw NPE when passed null 1297 //---------------------------------------------------------------- 1298 { 1299 clear(c); 1300 try { 1301 c.removeAll(null); 1302 fail("Expected NullPointerException"); 1303 } 1304 catch (NullPointerException e) { pass(); } 1305 catch (Throwable t) { unexpected(t); } 1306 1307 oneElement(c); 1308 try { 1309 c.removeAll(null); 1310 fail("Expected NullPointerException"); 1311 } 1312 catch (NullPointerException e) { pass(); } 1313 catch (Throwable t) { unexpected(t); } 1314 1315 clear(c); 1316 try { 1317 c.retainAll(null); 1318 fail("Expected NullPointerException"); 1319 } 1320 catch (NullPointerException e) { pass(); } 1321 catch (Throwable t) { unexpected(t); } 1322 1323 oneElement(c); 1324 try { 1325 c.retainAll(null); 1326 fail("Expected NullPointerException"); 1327 } 1328 catch (NullPointerException e) { pass(); } 1329 catch (Throwable t) { unexpected(t); } 1330 1331 oneElement(c); 1332 try { 1333 c.addAll(null); 1334 fail("Expected NullPointerException"); 1335 } 1336 catch (NullPointerException e) { pass(); } 1337 catch (Throwable t) { unexpected(t); } 1338 1339 oneElement(c); 1340 try { 1341 c.containsAll(null); 1342 fail("Expected NullPointerException"); 1343 } 1344 catch (NullPointerException e) { pass(); } 1345 catch (Throwable t) { unexpected(t); } 1346 } 1347 } 1348 1349 //---------------------------------------------------------------- 1350 // Map 1351 //---------------------------------------------------------------- 1352 private static void checkFunctionalInvariants(Map<Integer,Integer> m) { 1353 check(m.keySet().size() == m.entrySet().size()); 1354 check(m.keySet().size() == m.size()); 1355 checkFunctionalInvariants(m.keySet()); 1356 checkFunctionalInvariants(m.values()); 1357 check(m.size() != 0 ^ m.isEmpty()); 1358 check(! m.containsKey(ABSENT_VALUE)); 1359 1360 if (m instanceof Serializable) { 1361 //System.out.printf("Serializing %s%n", m.getClass().getName()); 1362 try { 1363 Object clone = serialClone(m); 1364 equal(m instanceof Serializable, 1365 clone instanceof Serializable); 1366 equal(m, clone); 1367 } catch (Error xxx) { 1368 if (! (xxx.getCause() instanceof NotSerializableException)) 1369 throw xxx; 1370 } 1371 } 1372 } 1373 1374 private static void testMap(Map<Integer,Integer> m) { 1375 System.out.println("\n==> " + m.getClass().getName()); 1376 1377 int hashCode = 0; 1378 for (var e : m.entrySet()) { 1379 int entryHash = (e.getKey() == null ? 0 : e.getKey().hashCode()) ^ 1380 (e.getValue() == null ? 0 : e.getValue().hashCode()); 1381 check(e.hashCode() == entryHash); 1382 hashCode += entryHash; 1383 } 1384 check(m.hashCode() == hashCode); 1385 1386 if (m instanceof ConcurrentMap) 1387 testConcurrentMap((ConcurrentMap<Integer,Integer>) m); 1388 1389 if (m instanceof NavigableMap) { 1390 System.out.println("NavigableMap tests..."); 1391 1392 NavigableMap<Integer,Integer> nm = 1393 (NavigableMap<Integer,Integer>) m; 1394 testNavigableMapRemovers(nm); 1395 testNavigableMap(nm); 1396 testNavigableMap(nm.headMap(6, false)); 1397 testNavigableMap(nm.headMap(5, true)); 1398 testNavigableMap(nm.tailMap(0, false)); 1399 testNavigableMap(nm.tailMap(1, true)); 1400 testNavigableMap(nm.subMap(1, true, 6, false)); 1401 testNavigableMap(nm.subMap(0, false, 5, true)); 1402 } 1403 1404 checkFunctionalInvariants(m); 1405 1406 if (supportsClear(m)) { 1407 try { clear(m); } 1408 catch (Throwable t) { unexpected(t); } 1409 } 1410 1411 if (supportsPut(m)) { 1412 try { 1413 check(m.put(3333, 77777) == null); 1414 check(m.put(9134, 74982) == null); 1415 check(m.get(9134) == 74982); 1416 check(m.put(9134, 1382) == 74982); 1417 check(m.get(9134) == 1382); 1418 check(m.size() == 2); 1419 checkFunctionalInvariants(m); 1420 checkNPEConsistency(m); 1421 } 1422 catch (Throwable t) { unexpected(t); } 1423 } 1424 } 1425 1426 private static boolean supportsPut(Map<Integer,Integer> m) { 1427 // We're asking for .equals(...) semantics 1428 if (m instanceof IdentityHashMap) return false; 1429 1430 try { check(m.put(ABSENT_VALUE,12735) == null); } 1431 catch (UnsupportedOperationException t) { return false; } 1432 catch (Throwable t) { unexpected(t); } 1433 1434 try { 1435 check(m.containsKey(ABSENT_VALUE)); 1436 check(m.remove(ABSENT_VALUE) != null); 1437 } catch (Throwable t) { unexpected(t); } 1438 return true; 1439 } 1440 1441 private static boolean supportsClear(Map<?,?> m) { 1442 try { m.clear(); } 1443 catch (UnsupportedOperationException t) { return false; } 1444 catch (Throwable t) { unexpected(t); } 1445 return true; 1446 } 1447 1448 //---------------------------------------------------------------- 1449 // ConcurrentMap 1450 //---------------------------------------------------------------- 1451 private static void testConcurrentMap(ConcurrentMap<Integer,Integer> m) { 1452 System.out.println("ConcurrentMap tests..."); 1453 1454 try { 1455 clear(m); 1456 1457 check(m.putIfAbsent(18357,7346) == null); 1458 check(m.containsKey(18357)); 1459 check(m.putIfAbsent(18357,8263) == 7346); 1460 try { m.putIfAbsent(18357,null); fail("NPE"); } 1461 catch (NullPointerException t) { } 1462 check(m.containsKey(18357)); 1463 1464 check(! m.replace(18357,8888,7777)); 1465 check(m.containsKey(18357)); 1466 try { m.replace(18357,null,7777); fail("NPE"); } 1467 catch (NullPointerException t) { } 1468 check(m.containsKey(18357)); 1469 check(m.get(18357) == 7346); 1470 check(m.replace(18357,7346,5555)); 1471 check(m.replace(18357,5555,7346)); 1472 check(m.get(18357) == 7346); 1473 1474 check(m.replace(92347,7834) == null); 1475 try { m.replace(18357,null); fail("NPE"); } 1476 catch (NullPointerException t) { } 1477 check(m.replace(18357,7346) == 7346); 1478 check(m.replace(18357,5555) == 7346); 1479 check(m.get(18357) == 5555); 1480 check(m.replace(18357,7346) == 5555); 1481 check(m.get(18357) == 7346); 1482 1483 check(! m.remove(18357,9999)); 1484 check(m.get(18357) == 7346); 1485 check(m.containsKey(18357)); 1486 check(! m.remove(18357,null)); // 6272521 1487 check(m.get(18357) == 7346); 1488 check(m.remove(18357,7346)); 1489 check(m.get(18357) == null); 1490 check(! m.containsKey(18357)); 1491 check(m.isEmpty()); 1492 1493 m.putIfAbsent(1,2); 1494 check(m.size() == 1); 1495 check(! m.remove(1,null)); 1496 check(! m.remove(1,null)); 1497 check(! m.remove(1,1)); 1498 check(m.remove(1,2)); 1499 check(m.isEmpty()); 1500 1501 testEmptyMap(m); 1502 } 1503 catch (Throwable t) { unexpected(t); } 1504 } 1505 1506 private static void throwsConsistently(Class<? extends Throwable> k, 1507 Iterable<Fun> fs) { 1508 List<Class<? extends Throwable>> threw = new ArrayList<>(); 1509 for (Fun f : fs) 1510 try { f.f(); threw.add(null); } 1511 catch (Throwable t) { 1512 check(k.isAssignableFrom(t.getClass())); 1513 threw.add(t.getClass()); 1514 } 1515 if (new HashSet<Object>(threw).size() != 1) 1516 fail(threw.toString()); 1517 } 1518 1519 private static <T> void checkNPEConsistency(final Map<T,Integer> m) { 1520 m.clear(); 1521 final ConcurrentMap<T,Integer> cm = (m instanceof ConcurrentMap) 1522 ? (ConcurrentMap<T,Integer>) m 1523 : null; 1524 List<Fun> fs = new ArrayList<>(); 1525 fs.add(() -> check(! m.containsKey(null))); 1526 fs.add(() -> equal(m.remove(null), null)); 1527 fs.add(() -> equal(m.get(null), null)); 1528 if (cm != null) 1529 fs.add(() -> check(! cm.remove(null,null))); 1530 throwsConsistently(NullPointerException.class, fs); 1531 1532 fs.clear(); 1533 final Map<T,Integer> sm = singletonMap(null,1); 1534 fs.add(() -> { equal(m.put(null,1), null); m.clear();}); 1535 fs.add(() -> { m.putAll(sm); m.clear();}); 1536 if (cm != null) { 1537 fs.add(() -> check(! cm.remove(null,null))); 1538 fs.add(() -> equal(cm.putIfAbsent(null,1), 1)); 1539 fs.add(() -> equal(cm.replace(null,1), null)); 1540 fs.add(() -> equal(cm.replace(null,1, 1), 1)); 1541 } 1542 throwsConsistently(NullPointerException.class, fs); 1543 } 1544 1545 //---------------------------------------------------------------- 1546 // NavigableMap 1547 //---------------------------------------------------------------- 1548 private static void 1549 checkNavigableMapKeys(NavigableMap<Integer,Integer> m, 1550 Integer i, 1551 Integer lower, 1552 Integer floor, 1553 Integer ceiling, 1554 Integer higher) { 1555 equal(m.lowerKey(i), lower); 1556 equal(m.floorKey(i), floor); 1557 equal(m.ceilingKey(i), ceiling); 1558 equal(m.higherKey(i), higher); 1559 } 1560 1561 private static void 1562 checkNavigableSetKeys(NavigableSet<Integer> m, 1563 Integer i, 1564 Integer lower, 1565 Integer floor, 1566 Integer ceiling, 1567 Integer higher) { 1568 equal(m.lower(i), lower); 1569 equal(m.floor(i), floor); 1570 equal(m.ceiling(i), ceiling); 1571 equal(m.higher(i), higher); 1572 } 1573 1574 static final Random rnd = new Random(); 1575 static void equalNext(final Iterator<?> it, Object expected) { 1576 if (rnd.nextBoolean()) 1577 check(it.hasNext()); 1578 equal(it.next(), expected); 1579 } 1580 1581 static void equalMaps(Map m1, Map m2) { 1582 equal(m1, m2); 1583 equal(m2, m1); 1584 equal(m1.size(), m2.size()); 1585 equal(m1.isEmpty(), m2.isEmpty()); 1586 equal(m1.toString(), m2.toString()); 1587 check(Arrays.equals(m1.entrySet().toArray(), m2.entrySet().toArray())); 1588 } 1589 1590 @SuppressWarnings({"unchecked", "rawtypes"}) 1591 static void testNavigableMapRemovers(NavigableMap m) 1592 { 1593 final Map emptyMap = new HashMap(); 1594 1595 final Map singletonMap = new HashMap(); 1596 singletonMap.put(1, 2); 1597 1598 abstract class NavigableMapView { 1599 abstract NavigableMap view(NavigableMap m); 1600 } 1601 1602 NavigableMapView[] views = { 1603 new NavigableMapView() { NavigableMap view(NavigableMap m) { 1604 return m; }}, 1605 new NavigableMapView() { NavigableMap view(NavigableMap m) { 1606 return m.headMap(99, true); }}, 1607 new NavigableMapView() { NavigableMap view(NavigableMap m) { 1608 return m.tailMap(-99, false); }}, 1609 new NavigableMapView() { NavigableMap view(NavigableMap m) { 1610 return m.subMap(-99, true, 99, false); }}, 1611 }; 1612 1613 abstract class Remover { 1614 abstract void remove(NavigableMap m, Object k, Object v); 1615 } 1616 1617 Remover[] removers = { 1618 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1619 equal(m.remove(k), v); }}, 1620 1621 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1622 equal(m.descendingMap().remove(k), v); }}, 1623 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1624 equal(m.descendingMap().headMap(-86, false).remove(k), v); }}, 1625 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1626 equal(m.descendingMap().tailMap(86, true).remove(k), v); }}, 1627 1628 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1629 equal(m.headMap(86, true).remove(k), v); }}, 1630 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1631 equal(m.tailMap(-86, true).remove(k), v); }}, 1632 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1633 equal(m.subMap(-86, false, 86, true).remove(k), v); }}, 1634 1635 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1636 check(m.keySet().remove(k)); }}, 1637 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1638 check(m.navigableKeySet().remove(k)); }}, 1639 1640 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1641 check(m.navigableKeySet().headSet(86, true).remove(k)); }}, 1642 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1643 check(m.navigableKeySet().tailSet(-86, false).remove(k)); }}, 1644 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1645 check(m.navigableKeySet().subSet(-86, true, 86, false) 1646 .remove(k)); }}, 1647 1648 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1649 check(m.descendingKeySet().headSet(-86, false).remove(k)); }}, 1650 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1651 check(m.descendingKeySet().tailSet(86, true).remove(k)); }}, 1652 new Remover() { void remove(NavigableMap m, Object k, Object v) { 1653 check(m.descendingKeySet().subSet(86, true, -86, false) 1654 .remove(k)); }}, 1655 }; 1656 1657 for (NavigableMapView view : views) { 1658 for (Remover remover : removers) { 1659 try { 1660 m.clear(); 1661 equalMaps(m, emptyMap); 1662 equal(m.put(1, 2), null); 1663 equalMaps(m, singletonMap); 1664 NavigableMap v = view.view(m); 1665 remover.remove(v, 1, 2); 1666 equalMaps(m, emptyMap); 1667 } catch (Throwable t) { unexpected(t); } 1668 } 1669 } 1670 } 1671 1672 private static void testNavigableMap(NavigableMap<Integer,Integer> m) 1673 { 1674 clear(m); 1675 checkNavigableMapKeys(m, 1, null, null, null, null); 1676 1677 equal(m.put(1, 2), null); 1678 equal(m.put(3, 4), null); 1679 equal(m.put(5, 9), null); 1680 1681 equal(m.put(1, 2), 2); 1682 equal(m.put(3, 4), 4); 1683 equal(m.put(5, 6), 9); 1684 1685 checkNavigableMapKeys(m, 0, null, null, 1, 1); 1686 checkNavigableMapKeys(m, 1, null, 1, 1, 3); 1687 checkNavigableMapKeys(m, 2, 1, 1, 3, 3); 1688 checkNavigableMapKeys(m, 3, 1, 3, 3, 5); 1689 checkNavigableMapKeys(m, 5, 3, 5, 5, null); 1690 checkNavigableMapKeys(m, 6, 5, 5, null, null); 1691 1692 for (final Iterator<Integer> it : 1693 (Iterator<Integer>[]) 1694 new Iterator<?>[] { 1695 m.descendingKeySet().iterator(), 1696 m.navigableKeySet().descendingIterator()}) { 1697 equalNext(it, 5); 1698 equalNext(it, 3); 1699 equalNext(it, 1); 1700 check(! it.hasNext()); 1701 THROWS(NoSuchElementException.class, () -> it.next()); 1702 } 1703 1704 { 1705 final Iterator<Map.Entry<Integer,Integer>> it 1706 = m.descendingMap().entrySet().iterator(); 1707 check(it.hasNext()); equal(it.next().getKey(), 5); 1708 check(it.hasNext()); equal(it.next().getKey(), 3); 1709 check(it.hasNext()); equal(it.next().getKey(), 1); 1710 check(! it.hasNext()); 1711 THROWS(NoSuchElementException.class, () -> it.next()); 1712 } 1713 1714 prepMapForDescItrTests(m); 1715 checkDescItrRmFirst(m.keySet(), m.navigableKeySet().descendingIterator()); 1716 prepMapForDescItrTests(m); 1717 checkDescItrRmMid(m.keySet(), m.navigableKeySet().descendingIterator()); 1718 prepMapForDescItrTests(m); 1719 checkDescItrRmLast(m.keySet(), m.navigableKeySet().descendingIterator()); 1720 1721 prepMapForDescItrTests(m); 1722 checkDescItrRmFirst(m.keySet(), m.descendingMap().keySet().iterator()); 1723 prepMapForDescItrTests(m); 1724 checkDescItrRmMid(m.keySet(), m.descendingMap().keySet().iterator()); 1725 prepMapForDescItrTests(m); 1726 checkDescItrRmLast(m.keySet(), m.descendingMap().keySet().iterator()); 1727 1728 prepMapForDescItrTests(m); 1729 checkDescItrRmFirst(m.keySet(), m.descendingKeySet().iterator()); 1730 prepMapForDescItrTests(m); 1731 checkDescItrRmMid(m.keySet(), m.descendingKeySet().iterator()); 1732 prepMapForDescItrTests(m); 1733 checkDescItrRmLast(m.keySet(), m.descendingKeySet().iterator()); 1734 1735 prepMapForDescItrTests(m); 1736 checkDescItrRmFirst(m.values(), m.descendingMap().values().iterator()); 1737 prepMapForDescItrTests(m); 1738 checkDescItrRmMid(m.values(), m.descendingMap().values().iterator()); 1739 prepMapForDescItrTests(m); 1740 checkDescItrRmLast(m.values(), m.descendingMap().values().iterator()); 1741 1742 prepMapForDescItrTests(m); 1743 checkDescItrRmFirst((Collection)m.entrySet(), 1744 m.descendingMap().entrySet().iterator()); 1745 prepMapForDescItrTests(m); 1746 checkDescItrRmMid((Collection)m.entrySet(), 1747 m.descendingMap().entrySet().iterator()); 1748 prepMapForDescItrTests(m); 1749 checkDescItrRmLast((Collection)m.entrySet(), 1750 m.descendingMap().entrySet().iterator()); 1751 } 1752 1753 private static void testNavigableSet(NavigableSet<Integer> s) { 1754 clear(s); 1755 checkNavigableSetKeys(s, 1, null, null, null, null); 1756 1757 check(s.add(1)); 1758 check(s.add(3)); 1759 check(s.add(5)); 1760 1761 check(! s.add(1)); 1762 check(! s.add(3)); 1763 check(! s.add(5)); 1764 1765 checkNavigableSetKeys(s, 0, null, null, 1, 1); 1766 checkNavigableSetKeys(s, 1, null, 1, 1, 3); 1767 checkNavigableSetKeys(s, 2, 1, 1, 3, 3); 1768 checkNavigableSetKeys(s, 3, 1, 3, 3, 5); 1769 checkNavigableSetKeys(s, 5, 3, 5, 5, null); 1770 checkNavigableSetKeys(s, 6, 5, 5, null, null); 1771 1772 for (final Iterator<Integer> it : 1773 (Iterator<Integer>[]) 1774 new Iterator<?>[] { 1775 s.descendingIterator(), 1776 s.descendingSet().iterator()}) { 1777 equalNext(it, 5); 1778 equalNext(it, 3); 1779 equalNext(it, 1); 1780 check(! it.hasNext()); 1781 THROWS(NoSuchElementException.class, () -> it.next()); 1782 } 1783 1784 prepSetForDescItrTests(s); 1785 checkDescItrRmFirst(s, s.descendingIterator()); 1786 prepSetForDescItrTests(s); 1787 checkDescItrRmMid(s, s.descendingIterator()); 1788 prepSetForDescItrTests(s); 1789 checkDescItrRmLast(s, s.descendingIterator()); 1790 1791 prepSetForDescItrTests(s); 1792 checkDescItrRmFirst(s, s.descendingSet().iterator()); 1793 prepSetForDescItrTests(s); 1794 checkDescItrRmMid(s, s.descendingSet().iterator()); 1795 prepSetForDescItrTests(s); 1796 checkDescItrRmLast(s, s.descendingSet().iterator()); 1797 } 1798 1799 private static void prepSetForDescItrTests(Set s) { 1800 clear(s); 1801 check(s.add(1)); 1802 check(s.add(3)); 1803 check(s.add(5)); 1804 } 1805 1806 private static void prepMapForDescItrTests(Map m) { 1807 clear(m); 1808 equal(m.put(1, 2), null); 1809 equal(m.put(3, 4), null); 1810 equal(m.put(5, 9), null); 1811 } 1812 1813 //-------------------------------------------------------------------- 1814 // Check behavior of descending iterator when first element is removed 1815 //-------------------------------------------------------------------- 1816 private static <T> void checkDescItrRmFirst(Collection<T> ascColl, 1817 Iterator<T> descItr) { 1818 T[] expected = (T[]) ascColl.toArray(); 1819 int idx = expected.length -1; 1820 1821 equalNext(descItr, expected[idx--]); 1822 descItr.remove(); 1823 while (idx >= 0 && descItr.hasNext()) { 1824 equalNext(descItr, expected[idx--]); 1825 } 1826 equal(descItr.hasNext(), false); 1827 equal(idx, -1); 1828 } 1829 1830 //----------------------------------------------------------------------- 1831 // Check behavior of descending iterator when a middle element is removed 1832 //----------------------------------------------------------------------- 1833 private static <T> void checkDescItrRmMid(Collection<T> ascColl, 1834 Iterator<T> descItr) { 1835 T[] expected = (T[]) ascColl.toArray(); 1836 int idx = expected.length -1; 1837 1838 while (idx >= expected.length / 2) { 1839 equalNext(descItr, expected[idx--]); 1840 } 1841 descItr.remove(); 1842 while (idx >= 0 && descItr.hasNext()) { 1843 equalNext(descItr, expected[idx--]); 1844 } 1845 equal(descItr.hasNext(), false); 1846 equal(idx, -1); 1847 } 1848 1849 //----------------------------------------------------------------------- 1850 // Check behavior of descending iterator when the last element is removed 1851 //----------------------------------------------------------------------- 1852 private static <T> void checkDescItrRmLast(Collection<T> ascColl, 1853 Iterator<T> descItr) { 1854 T[] expected = (T[]) ascColl.toArray(); 1855 int idx = expected.length -1; 1856 1857 while (idx >= 0 && descItr.hasNext()) { 1858 equalNext(descItr, expected[idx--]); 1859 } 1860 equal(idx, -1); 1861 equal(descItr.hasNext(), false); 1862 descItr.remove(); 1863 equal(ascColl.contains(expected[0]), false); 1864 } 1865 1866 //--------------------- Infrastructure --------------------------- 1867 static volatile int passed = 0, failed = 0; 1868 static void pass() { passed++; } 1869 static void fail() { failed++; Thread.dumpStack(); } 1870 static void fail(String msg) { System.out.println(msg); fail(); } 1871 static void unexpected(Throwable t) { failed++; t.printStackTrace(); } 1872 static void check(boolean cond) { if (cond) pass(); else fail(); } 1873 static void equal(Object x, Object y) { 1874 if (x == null ? y == null : x.equals(y)) pass(); 1875 else {System.out.println(x + " not equal to " + y); fail();}} 1876 static void equal(Object x, Object y, String msg) { 1877 if (x == null ? y == null : x.equals(y)) pass(); 1878 else {System.out.println(x + " not equal to " + y + " : " + msg); fail();}} 1879 static void equal2(Object x, Object y) {equal(x, y); equal(y, x);} 1880 public static void main(String[] args) throws Throwable { 1881 try { realMain(args); } catch (Throwable t) { unexpected(t); } 1882 1883 System.out.printf("%nPassed = %d, failed = %d%n%n", passed, failed); 1884 if (failed > 0) throw new Exception("Some tests failed"); 1885 } 1886 interface Fun {void f() throws Throwable;} 1887 private static void THROWS(Class<? extends Throwable> k, Fun... fs) { 1888 for (Fun f : fs) 1889 try { f.f(); fail("Expected " + k.getName() + " not thrown"); } 1890 catch (Throwable t) { 1891 if (k.isAssignableFrom(t.getClass())) pass(); 1892 else unexpected(t);}} 1893 static byte[] serializedForm(Object obj) { 1894 try { 1895 ByteArrayOutputStream baos = new ByteArrayOutputStream(); 1896 new ObjectOutputStream(baos).writeObject(obj); 1897 return baos.toByteArray(); 1898 } catch (IOException e) { throw new Error(e); }} 1899 static Object readObject(byte[] bytes) 1900 throws IOException, ClassNotFoundException { 1901 InputStream is = new ByteArrayInputStream(bytes); 1902 return new ObjectInputStream(is).readObject();} 1903 @SuppressWarnings("unchecked") 1904 static <T> T serialClone(T obj) { 1905 try { return (T) readObject(serializedForm(obj)); } 1906 catch (Exception e) { throw new Error(e); }} 1907 private static class NewAbstractCollection<E> extends AbstractCollection<E> { 1908 ArrayList<E> list = new ArrayList<>(); 1909 public boolean remove(Object obj) { 1910 return list.remove(obj); 1911 } 1912 public boolean add(E e) { 1913 return list.add(e); 1914 } 1915 public Iterator<E> iterator() { 1916 return list.iterator(); 1917 } 1918 public int size() { 1919 return list.size(); 1920 } 1921 } 1922 private static class NewAbstractSet<E> extends AbstractSet<E> { 1923 HashSet<E> set = new HashSet<>(); 1924 public boolean remove(Object obj) { 1925 return set.remove(obj); 1926 } 1927 public boolean add(E e) { 1928 return set.add(e); 1929 } 1930 public Iterator<E> iterator() { 1931 return set.iterator(); 1932 } 1933 public int size() { 1934 return set.size(); 1935 } 1936 } 1937 1938 } --- EOF ---