Reduce listener count

This commit is contained in:
TheMode 2021-08-22 03:08:37 +02:00
parent 6f88abf987
commit 9c7c409fae

View File

@ -327,8 +327,10 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
for (var mappedEntry : mappedNodeCache.entrySet()) { for (var mappedEntry : mappedNodeCache.entrySet()) {
final EventNodeImpl<E> mappedNode = mappedEntry.getValue(); final EventNodeImpl<E> mappedNode = mappedEntry.getValue();
if (!mappedNode.eventType.isAssignableFrom(eventType)) continue; if (!mappedNode.eventType.isAssignableFrom(eventType)) continue;
if (mappedNode.listenerMap.isEmpty()) final var listenerMap = mappedNode.listenerMap;
if (listenerMap.isEmpty())
continue; // The mapped node does not have any listener (perhaps throw a warning?) continue; // The mapped node does not have any listener (perhaps throw a warning?)
if (!listenerMap.containsKey(eventType)) continue;
filters.add(mappedNode.filter); filters.add(mappedNode.filter);
handlers.put(mappedEntry.getKey(), Pair.of(mappedNode, mappedNode.getHandle(eventType))); handlers.put(mappedEntry.getKey(), Pair.of(mappedNode, mappedNode.getHandle(eventType)));
} }
@ -366,25 +368,37 @@ class EventNodeImpl<T extends Event> implements EventNode<T> {
private void appendEntries(ListenerEntry<E> entry, EventNodeImpl<E> targetNode) { private void appendEntries(ListenerEntry<E> entry, EventNodeImpl<E> targetNode) {
final var filter = targetNode.filter; final var filter = targetNode.filter;
final var predicate = targetNode.predicate; final var predicate = targetNode.predicate;
// Normal listeners
for (var listener : entry.listeners) { final boolean hasPredicate = predicate != null;
if (predicate != null) { final var listenersCopy = List.copyOf(entry.listeners);
// Ensure that the event is valid before running final var bindingsCopy = List.copyOf(entry.bindingConsumers);
this.listeners.add(e -> { if (!hasPredicate && listenersCopy.isEmpty() && bindingsCopy.isEmpty())
final var value = filter.getHandler(e); return; // Nothing to run
if (!predicate.test(e, value)) return;
callListener(targetNode, listener, e); if (!hasPredicate && bindingsCopy.isEmpty() && listenersCopy.size() == 1) {
}); // Only one normal listener
} else { final EventListener<E> listener = listenersCopy.get(0);
// No predicate, run directly this.listeners.add(e -> callListener(targetNode, listener, e));
this.listeners.add(e -> callListener(targetNode, listener, e)); return;
}
// Worth case scenario, try to run everything
this.listeners.add(e -> {
if (hasPredicate) {
final var value = filter.getHandler(e);
if (!predicate.test(e, value)) return;
} }
} if (!listenersCopy.isEmpty()) {
// Bindings for (EventListener<E> listener : listenersCopy) {
final var bindingConsumers = entry.bindingConsumers; callListener(targetNode, listener, e);
if (!bindingConsumers.isEmpty()) { // Ensure no array clone }
this.listeners.addAll(bindingConsumers); }
} if (!bindingsCopy.isEmpty()) {
for (Consumer<E> eConsumer : bindingsCopy) {
eConsumer.accept(e);
}
}
});
} }
static <E extends Event> void callListener(EventNodeImpl<E> targetNode, EventListener<E> listener, E event) { static <E extends Event> void callListener(EventNodeImpl<E> targetNode, EventListener<E> listener, E event) {