/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions.
*/
/* * This file is available under and governed by the GNU General Public * License version 2 only, as published by the Free Software Foundation. * However, the following notice accompanied the original version of this * file: * * Written by Doug Lea and Martin Buchholz with assistance from * members of JCP JSR-166 Expert Group and released to the public * domain, as explained at * http://creativecommons.org/publicdomain/zero/1.0/
*/
publicvoid testImplSanity() { final ThreadLocalRandom rnd = ThreadLocalRandom.current();
{
Map m = impl.emptyMap();
assertTrue(m.isEmpty());
mustEqual(0, m.size());
Object k = impl.makeKey(rnd.nextInt());
Object v = impl.makeValue(rnd.nextInt());
m.put(k, v);
assertFalse(m.isEmpty());
mustEqual(1, m.size());
assertTrue(m.containsKey(k));
assertTrue(m.containsValue(v));
}
{
Map m = impl.emptyMap();
Object v = impl.makeValue(rnd.nextInt()); if (impl.permitsNullKeys()) {
m.put(null, v);
assertTrue(m.containsKey(null));
assertTrue(m.containsValue(v));
} else {
assertThrows(NullPointerException.class, () -> m.put(null, v));
}
}
{
Map m = impl.emptyMap();
Object k = impl.makeKey(rnd.nextInt()); if (impl.permitsNullValues()) {
m.put(k, null);
assertTrue(m.containsKey(k));
assertTrue(m.containsValue(null));
} else {
assertThrows(NullPointerException.class, () -> m.put(k, null));
}
}
{
Map m = impl.emptyMap();
Object k = impl.makeKey(rnd.nextInt());
Object v1 = impl.makeValue(rnd.nextInt());
Object v2 = impl.makeValue(rnd.nextInt());
m.put(k, v1); if (impl.supportsSetValue()) {
((Map.Entry)(m.entrySet().iterator().next())).setValue(v2);
assertSame(v2, m.get(k));
assertTrue(m.containsKey(k));
assertTrue(m.containsValue(v2));
assertFalse(m.containsValue(v1));
} else {
assertThrows(UnsupportedOperationException.class,
() -> ((Map.Entry)(m.entrySet().iterator().next())).setValue(v2));
}
}
}
/** * Tests and extends the scenario reported in * https://bugs.openjdk.org/browse/JDK-8186171 * HashMap: Entry.setValue may not work after Iterator.remove() called for previous entries * ant -Djsr166.tckTestClass=HashMapTest -Djsr166.methodFilter=testBug8186171 -Djsr166.runsPerTest=1000 tck
*/ publicvoid testBug8186171() { if (!impl.supportsSetValue()) return; if (!atLeastJava10()) return; // jdk9 is no longer maintained final ThreadLocalRandom rnd = ThreadLocalRandom.current(); finalboolean permitsNullValues = impl.permitsNullValues(); final Object v1 = (permitsNullValues && rnd.nextBoolean())
? null : impl.makeValue(1); final Object v2 = (permitsNullValues && rnd.nextBoolean() && v1 != null)
? null : impl.makeValue(2);
// If true, always lands in first bucket in hash tables. finalboolean poorHash = rnd.nextBoolean(); class Key implements Comparable<Key> { finalint i;
Key(int i) { this.i = i; } publicint hashCode() { return poorHash ? 0 : super.hashCode(); } publicint compareTo(Key x) { return Integer.compare(this.i, x.i);
}
}
// Both HashMap and ConcurrentHashMap have: // TREEIFY_THRESHOLD = 8; UNTREEIFY_THRESHOLD = 6; finalint size = rnd.nextInt(1, 25);
List<Key> keys = new ArrayList<>(); for (int i = size; i-->0; ) keys.add(new Key(i));
Key keyToFrob = keys.get(rnd.nextInt(keys.size()));
Map<Key, Object> m = impl.emptyMap(); for (Key key : keys) m.put(key, v1);
for (Iterator<Map.Entry<Key, Object>> it = m.entrySet().iterator();
it.hasNext(); ) {
Map.Entry<Key, Object> entry = it.next(); if (entry.getKey() == keyToFrob)
entry.setValue(v2); // does this have the expected effect? else
it.remove();
}
/** * "Missing" test found while investigating JDK-8210280. * ant -Djsr166.tckTestClass=HashMapTest -Djsr166.methodFilter=testBug8210280 -Djsr166.runsPerTest=1000000 tck
*/ publicvoid testBug8210280() { final ThreadLocalRandom rnd = ThreadLocalRandom.current(); finalint size1 = rnd.nextInt(32); finalint size2 = rnd.nextInt(128);
final Map m1 = impl.emptyMap(); for (int i = 0; i < size1; i++) { int elt = rnd.nextInt(1024 * i, 1024 * (i + 1));
assertNull(m1.put(impl.makeKey(elt), impl.makeValue(elt)));
}
final Map m2 = impl.emptyMap(); for (int i = 0; i < size2; i++) { int elt = rnd.nextInt(Integer.MIN_VALUE + 1024 * i,
Integer.MIN_VALUE + 1024 * (i + 1));
assertNull(m2.put(impl.makeKey(elt), impl.makeValue(-elt)));
}
final Map m1Copy = impl.emptyMap();
m1Copy.putAll(m1);
m1.putAll(m2);
for (Object elt : m2.keySet())
mustEqual(m2.get(elt), m1.get(elt)); for (Object elt : m1Copy.keySet())
assertSame(m1Copy.get(elt), m1.get(elt));
mustEqual(size1 + size2, m1.size());
}
/** * 8222930: ConcurrentSkipListMap.clone() shares size variable between original and clone
*/ publicvoid testClone() { final ThreadLocalRandom rnd = ThreadLocalRandom.current(); finalint size = rnd.nextInt(4); final Map map = impl.emptyMap(); for (int i = 0; i < size; i++)
map.put(impl.makeKey(i), impl.makeValue(i)); final Map clone = cloneableClone(map); if (clone == null) return; // not cloneable?
/** * Concurrent access by compute methods behaves as expected
*/ publicvoid testConcurrentAccess() throws Throwable { final Map map = impl.emptyMap(); finallong testDurationMillis = expensiveTests ? 1000 : 2; finalint nTasks = impl.isConcurrent()
? ThreadLocalRandom.current().nextInt(1, 10)
: 1; final AtomicBoolean done = new AtomicBoolean(false); finalboolean remappingFunctionCalledAtMostOnce
= impl.remappingFunctionCalledAtMostOnce(); final List<CompletableFuture> futures = new ArrayList<>(); final AtomicLong expectedSum = new AtomicLong(0); final Action[] tasks = { // repeatedly increment values using compute()
() -> { long[] invocations = newlong[2];
ThreadLocalRandom rnd = ThreadLocalRandom.current();
BiFunction<Object, Object, Object> incValue = (k, v) -> {
invocations[1]++; int vi = (v == null) ? 1 : impl.valueToInt(v) + 1; return impl.makeValue(vi);
}; while (!done.getAcquire()) {
invocations[0]++;
Object key = impl.makeKey(3 * rnd.nextInt(10));
map.compute(key, incValue);
} if (remappingFunctionCalledAtMostOnce)
mustEqual(invocations[0], invocations[1]);
expectedSum.getAndAdd(invocations[0]);
}, // repeatedly increment values using computeIfPresent()
() -> { long[] invocations = newlong[2];
ThreadLocalRandom rnd = ThreadLocalRandom.current();
BiFunction<Object, Object, Object> incValue = (k, v) -> {
invocations[1]++; int vi = impl.valueToInt(v) + 1; return impl.makeValue(vi);
}; while (!done.getAcquire()) {
Object key = impl.makeKey(3 * rnd.nextInt(10)); if (map.computeIfPresent(key, incValue) != null)
invocations[0]++;
} if (remappingFunctionCalledAtMostOnce)
mustEqual(invocations[0], invocations[1]);
expectedSum.getAndAdd(invocations[0]);
},
}; for (int i = nTasks; i--> 0; ) {
Action task = chooseRandomly(tasks);
futures.add(CompletableFuture.runAsync(checkedRunnable(task)));
} Thread.sleep(testDurationMillis);
done.setRelease(true); for (var future : futures)
checkTimedGet(future, null);
long sum = map.values().stream().mapToLong(x -> (int) x).sum();
mustEqual(expectedSum.get(), sum);
}
// public void testFailsIntentionallyForDebugging() { // fail(impl.klazz().getSimpleName()); // }
}
¤ Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.0.18Bemerkung:
(vorverarbeitet)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung ist noch experimentell.