mirror of
https://github.com/corda/corda.git
synced 2025-01-07 13:38:47 +00:00
c0d178d5f1
This is the simplest possible ConcurrentHashMap I could come up with that works and is actually concurrent in the way one would expect. It's pretty unconventional, being based on a persistent red-black tree, and not particularly memory-efficient or cache-friendly. I think this is a good place to start, though, and it should perform reasonably well for most workloads. Patches for a more efficient implementation are welcome! I also implemented AtomicReferenceArray, since I was using it in my first, naive attempt to implement ConcurrentHashMap. I had to do a bit of refactoring, including moving some non-standard stuff from java.util.Collections to avian.Data so I could make it available to code outside the java.util package, which is why I had to modify several unrelated files.
167 lines
4.7 KiB
Java
167 lines
4.7 KiB
Java
import java.util.Iterator;
|
|
import java.util.Map;
|
|
import java.util.concurrent.ConcurrentMap;
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
|
|
public class Concurrent {
|
|
private static final int ThreadCount = 4;
|
|
private static final int IterationCount = 100;
|
|
private static final int Range = 10;
|
|
private static final int CommonBase = -Range;
|
|
|
|
private static void expect(boolean v) {
|
|
if (! v) throw new RuntimeException();
|
|
}
|
|
|
|
public static void main(String[] args) throws Throwable {
|
|
final ConcurrentMap<Integer, Object> map = new ConcurrentHashMap();
|
|
final int[] counter = new int[1];
|
|
final int[] step = new int[1];
|
|
final Throwable[] exception = new Throwable[1];
|
|
|
|
synchronized (map) {
|
|
for (int i = 0; i < ThreadCount; ++i) {
|
|
final int index = i;
|
|
new Thread() {
|
|
public void run() {
|
|
try {
|
|
synchronized (map) {
|
|
++ counter[0];
|
|
map.notifyAll();
|
|
while (exception[0] == null && step[0] == 0) {
|
|
map.wait();
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < IterationCount; ++i) {
|
|
populateCommon(map);
|
|
populate(map, index * Range);
|
|
}
|
|
|
|
synchronized (map) {
|
|
-- counter[0];
|
|
map.notifyAll();
|
|
while (exception[0] == null && step[0] == 1) {
|
|
map.wait();
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < IterationCount; ++i) {
|
|
populate(map, index * Range);
|
|
depopulate(map, index * Range);
|
|
}
|
|
|
|
synchronized (map) {
|
|
++ counter[0];
|
|
map.notifyAll();
|
|
}
|
|
} catch (Throwable e) {
|
|
synchronized (map) {
|
|
exception[0] = e;
|
|
map.notifyAll();
|
|
}
|
|
e.printStackTrace();
|
|
}
|
|
}
|
|
}.start();
|
|
}
|
|
|
|
try {
|
|
while (exception[0] == null && counter[0] < ThreadCount) {
|
|
map.wait();
|
|
}
|
|
|
|
step[0] = 1;
|
|
map.notifyAll();
|
|
|
|
while (exception[0] == null && counter[0] > 0) {
|
|
map.wait();
|
|
}
|
|
|
|
if (map.size() != ThreadCount * Range) {
|
|
System.err.println
|
|
("expected " + (ThreadCount * Range) + " got " + map.size());
|
|
}
|
|
expect(map.size() == ThreadCount * Range);
|
|
for (int i = CommonBase, j = CommonBase + Range; i < j; ++i) {
|
|
expect(! map.containsKey(i));
|
|
}
|
|
|
|
step[0] = 2;
|
|
map.notifyAll();
|
|
|
|
while (exception[0] == null && counter[0] < ThreadCount) {
|
|
map.wait();
|
|
}
|
|
|
|
expect(map.isEmpty());
|
|
expect(exception[0] == null);
|
|
} catch (Throwable e) {
|
|
exception[0] = e;
|
|
throw e;
|
|
} finally {
|
|
map.notifyAll();
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void populateCommon(ConcurrentMap<Integer, Object> map) {
|
|
Object value = new Object();
|
|
for (int i = CommonBase, j = CommonBase + Range; i < j; ++i) {
|
|
map.remove(i);
|
|
map.put(i, value);
|
|
map.remove(i);
|
|
}
|
|
}
|
|
|
|
private static void populate(ConcurrentMap<Integer, Object> map, int base) {
|
|
for (int i = base, j = base + Range; i < j; ++i) {
|
|
map.remove(i);
|
|
Object value = new Object();
|
|
expect(map.put(i, value) == null);
|
|
expect(map.containsKey(i));
|
|
expect(map.get(i).equals(value));
|
|
expect(map.putIfAbsent(i, new Object()) == value);
|
|
expect(map.get(i).equals(value));
|
|
expect(! map.remove(i, new Object()));
|
|
expect(map.remove(i, value));
|
|
expect(map.replace(i, value) == null);
|
|
expect(! map.containsKey(i));
|
|
expect(map.get(i) == null);
|
|
expect(map.putIfAbsent(i, value) == null);
|
|
expect(map.containsKey(i));
|
|
expect(map.get(i) == value);
|
|
Object newValue = new Object();
|
|
expect(map.replace(i, newValue) == value);
|
|
expect(map.get(i) == newValue);
|
|
|
|
boolean found = false;
|
|
for (Iterator<Map.Entry<Integer, Object>> it = map.entrySet().iterator();
|
|
it.hasNext();)
|
|
{
|
|
Map.Entry<Integer, Object> e = it.next();
|
|
if (e.getKey() == i) {
|
|
expect(! found);
|
|
expect(e.getValue() == newValue);
|
|
found = true;
|
|
it.remove();
|
|
}
|
|
}
|
|
|
|
expect(found);
|
|
expect(! map.containsKey(i));
|
|
expect(map.putIfAbsent(i, value) == null);
|
|
expect(map.containsKey(i));
|
|
expect(map.get(i) == value);
|
|
}
|
|
}
|
|
|
|
private static void depopulate(ConcurrentMap<Integer, Object> map, int base)
|
|
{
|
|
for (int i = base, j = base + Range; i < j; ++i) {
|
|
expect(map.containsKey(i));
|
|
expect(map.remove(i) != null);
|
|
}
|
|
}
|
|
}
|