Изменить:
Я провел ваш код через набор тестов с десятью потоками, добавляющими и сравнивающими тысячи шаблонов, но я не нашел ничего плохого в вашей реализации. Я полагаю, что вы неверно истолковываете свои данные. Например, в многопоточной среде это иногда возвращает false:
// sometimes this can be false
blacklist.contains(pattern) == blacklist.contains(pattern);
Другой поток изменил черный список после первого вызова, но до второго вызова. Это нормальное поведение, и сам класс не может ничего сделать, чтобы остановить его. Если это не то поведение, которое вам нужно, вы можете синхронизировать его извне класса:
synchronized (blacklist) {
// this will always be true
blacklist.contains(pattern) == blacklist.contains(pattern);
}
Оригинальный ответ:
Вы синхронизируете корневой узел, но он не синхронизирует ни одного из его дочерних узлов. Все, что вам нужно сделать, чтобы сделать ваш класс пуленепробиваемым, это синхронизировать методы add(int[])
и contains(int[])
, а затем не пропускать ссылки. Это гарантирует, что только один поток может когда-либо использовать объект черного списка одновременно.
Я возился с вашим кодом, пытаясь разобраться в нем, так что вы могли бы также иметь его:
import java.util.HashMap;
import java.util.Map;
import java.util.Stack;
public class Blacklist {
private final Node root = new Node(Integer.MIN_VALUE, false);
public synchronized void add(int[] array) {
if (array == null) return;
Node next, cur = root;
for(int i = 0; i < array.length-1 && !cur.isLeaf(); i++) {
next = cur.getChild(array[i]);
if (next == null) {
next = new Node(array[i], false);
cur.addChild(next);
}
cur = next;
}
if (!cur.isLeaf()) {
next = cur.getChild(array[array.length-1]);
if (next == null || !next.isLeaf())
cur.addChild(new Node(array[array.length-1], true));
}
}
public synchronized boolean contains(int[] array) {
if (array == null) return false;
Node cur = root;
for (int i = 0; i < array.length; i++) {
cur = cur.getChild(array[i]);
if (cur == null) return false;
if (cur.isLeaf()) return true;
}
return false;
}
private static class Node {
private final Map<Integer, Node> children;
private final int value;
public Node(int _value, boolean leaf) {
children = (leaf?null:new HashMap<Integer, Node>());
value = _value;
}
public void addChild(Node child) { children.put(child.value, child); }
public Node getChild(int value) { return children.get(value); }
public boolean isLeaf() { return (children == null); }
}
}
Рамка Collections может упростить вам задачу. Вы не делаете себе никаких одолжений, переопределяя ArrayList.
Здесь я использую HashMap, чтобы вам не нужно было выделять более 9000 ссылок для чего-то вроде этого:
blacklist.add(new int[] {1, 2000, 3000, 4000});