Есть ли лучший способ иметь слушателя в java-коллекции, чем переносить его в класс, реализующий шаблон наблюдателя?
Хороший способ получить прослушиватель коллекции?
Ответ 1
Вы должны проверить Застекленные списки
Он содержит наблюдаемые классы List, которые запускают события, когда элементы добавляются, удаляются, заменяются и т.д.
Ответ 2
Вы можете использовать ForwardingSet, ForwardingList и т.д. из Guava, чтобы украсить конкретный экземпляр желаемым поведением.
Здесь моя собственная реализация, которая просто использует простые API JDK:
// create an abstract class that implements this interface with blank implementations
// that way, annonymous subclasses can observe only the events they care about
public interface CollectionObserver<E> {
public void beforeAdd(E o);
public void afterAdd(E o);
// other events to be observed ...
}
// this method would go in a utility class
public static <E> Collection<E> observedCollection(
final Collection<E> collection, final CollectionObserver<E> observer) {
return new Collection<E>() {
public boolean add(final E o) {
observer.beforeAdd(o);
boolean result = collection.add(o);
observer.afterAdd(o);
return result;
}
// ... generate rest of delegate methods in Eclipse
};
}
Ответ 3
"Commons-Events предоставляет дополнительные классы для запуска и обработки событий. Он фокусируется на структуре коллекций Java, предоставляя декораторам другим коллекциям, которые запускают события".
Ответ 4
Ну, если вам действительно не нужен экземпляр java.util.Collection или List, вы можете использовать DefaultListModel. Я не знаю никаких "реальных" реализаций коллекции со встроенной поддержкой слушателя/наблюдателя.
Ответ 5
есть много способов добиться этого - часто я использую этот подход
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
public class ObservableArrayList<E> extends ArrayList<E> {
private @interface MethodId {
private static final int REMOVE = 2;
private static final int ADD = 1;
}
public interface ListObserver<E> {
void onElementAdded(E element);
void onElementRemoved(E element);
}
public ObservableArrayList(int capacity) {
super(capacity);
ensureObserver();
}
public ObservableArrayList() {
ensureObserver();
}
public ObservableArrayList(Collection<? extends E> collection) {
super(collection);
ensureObserver();
}
private List<WeakReference<ListObserver<E>>> _listObserverWeakRefList;
public void addObserver(ListObserver<E> observer) {
_listObserverWeakRefList.add(new WeakReference<ListObserver<E>> (observer));
}
private void ensureObserver() {
if (_listObserverWeakRefList == null) {
_listObserverWeakRefList = new ArrayList<>();
}
}
@Override
public boolean add(E object) {
super.add(object);
callObservable(MethodId.ADD, object);
return true;
}
@Override
public boolean remove(Object object) {
boolean removed = super.remove(object);
if (removed) callObservable(MethodId.REMOVE, object);
return removed;
}
private void callObservable(@MethodId int methodId, Object element) {
for (WeakReference<ListObserver<E>> observerRef : _listObserverWeakRefList) {
ListObserver<E> observer = observerRef.get();
if (observer != null) {
switch (methodId) {
case MethodId.ADD:
observer.onElementAdded((E) element);
break;
case MethodId.REMOVE:
observer.onElementRemoved((E) element);
break;
}
}
}
}
}