Mirror von
https://github.com/IntellectualSites/FastAsyncWorldEdit.git
synchronisiert 2024-11-08 04:20:06 +01:00
Re-write EventBus to be faster
Dieser Commit ist enthalten in:
Ursprung
389671b43b
Commit
a2b67f8ddb
@ -1,33 +0,0 @@
|
|||||||
/*
|
|
||||||
* WorldEdit, a Minecraft world manipulation toolkit
|
|
||||||
* Copyright (C) sk89q <http://www.sk89q.com>
|
|
||||||
* Copyright (C) WorldEdit team and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as published by the
|
|
||||||
* Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program 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 Lesser General Public License
|
|
||||||
* for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.sk89q.worldedit.internal.annotation;
|
|
||||||
|
|
||||||
import java.lang.annotation.Documented;
|
|
||||||
import java.lang.annotation.Retention;
|
|
||||||
import java.lang.annotation.RetentionPolicy;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Marks features that should be replaced with Google Guava but cannot
|
|
||||||
* yet because Bukkit uses such an old version of Guava.
|
|
||||||
*/
|
|
||||||
@Retention(RetentionPolicy.SOURCE)
|
|
||||||
@Documented
|
|
||||||
public @interface RequiresNewerGuava {
|
|
||||||
}
|
|
@ -19,11 +19,9 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.util.eventbus;
|
package com.sk89q.worldedit.util.eventbus;
|
||||||
|
|
||||||
|
import com.google.common.collect.HashMultimap;
|
||||||
import com.google.common.collect.Multimap;
|
import com.google.common.collect.Multimap;
|
||||||
import com.google.common.collect.Multimaps;
|
|
||||||
import com.google.common.collect.SetMultimap;
|
import com.google.common.collect.SetMultimap;
|
||||||
import com.google.common.eventbus.DeadEvent;
|
|
||||||
import com.sk89q.worldedit.internal.annotation.RequiresNewerGuava;
|
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@ -31,11 +29,11 @@ import java.lang.reflect.InvocationTargetException;
|
|||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.locks.ReadWriteLock;
|
||||||
|
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
||||||
|
|
||||||
import static com.google.common.base.Preconditions.checkNotNull;
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
|
||||||
@ -46,17 +44,15 @@ import static com.google.common.base.Preconditions.checkNotNull;
|
|||||||
* <p>This class is based on Guava's {@link EventBus} but priority is supported
|
* <p>This class is based on Guava's {@link EventBus} but priority is supported
|
||||||
* and events are dispatched at the time of call, rather than being queued up.
|
* and events are dispatched at the time of call, rather than being queued up.
|
||||||
* This does allow dispatching during an in-progress dispatch.</p>
|
* This does allow dispatching during an in-progress dispatch.</p>
|
||||||
*
|
|
||||||
* <p>This implementation utilizes naive synchronization on all getter and
|
|
||||||
* setter methods. Dispatch does not occur when a lock has been acquired,
|
|
||||||
* however.</p>
|
|
||||||
*/
|
*/
|
||||||
public class EventBus {
|
public final class EventBus {
|
||||||
|
|
||||||
private final Logger logger = LoggerFactory.getLogger(EventBus.class);
|
private final Logger logger = LoggerFactory.getLogger(EventBus.class);
|
||||||
|
|
||||||
|
private final ReadWriteLock lock = new ReentrantReadWriteLock();
|
||||||
|
|
||||||
private final SetMultimap<Class<?>, EventHandler> handlersByType =
|
private final SetMultimap<Class<?>, EventHandler> handlersByType =
|
||||||
Multimaps.newSetMultimap(new HashMap<>(), this::newHandlerSet);
|
HashMultimap.create();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Strategy for finding handler methods in registered objects. Currently,
|
* Strategy for finding handler methods in registered objects. Currently,
|
||||||
@ -65,7 +61,6 @@ public class EventBus {
|
|||||||
*/
|
*/
|
||||||
private final SubscriberFindingStrategy finder = new AnnotatedSubscriberFinder();
|
private final SubscriberFindingStrategy finder = new AnnotatedSubscriberFinder();
|
||||||
|
|
||||||
@RequiresNewerGuava
|
|
||||||
private HierarchyCache flattenHierarchyCache = new HierarchyCache();
|
private HierarchyCache flattenHierarchyCache = new HierarchyCache();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -74,10 +69,15 @@ public class EventBus {
|
|||||||
* @param clazz the event class to register
|
* @param clazz the event class to register
|
||||||
* @param handler the handler to register
|
* @param handler the handler to register
|
||||||
*/
|
*/
|
||||||
public synchronized void subscribe(Class<?> clazz, EventHandler handler) {
|
public void subscribe(Class<?> clazz, EventHandler handler) {
|
||||||
checkNotNull(clazz);
|
checkNotNull(clazz);
|
||||||
checkNotNull(handler);
|
checkNotNull(handler);
|
||||||
|
lock.writeLock().lock();
|
||||||
|
try {
|
||||||
handlersByType.put(clazz, handler);
|
handlersByType.put(clazz, handler);
|
||||||
|
} finally {
|
||||||
|
lock.writeLock().unlock();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -85,9 +85,14 @@ public class EventBus {
|
|||||||
*
|
*
|
||||||
* @param handlers a map of handlers
|
* @param handlers a map of handlers
|
||||||
*/
|
*/
|
||||||
public synchronized void subscribeAll(Multimap<Class<?>, EventHandler> handlers) {
|
public void subscribeAll(Multimap<Class<?>, EventHandler> handlers) {
|
||||||
checkNotNull(handlers);
|
checkNotNull(handlers);
|
||||||
|
lock.writeLock().lock();
|
||||||
|
try {
|
||||||
handlersByType.putAll(handlers);
|
handlersByType.putAll(handlers);
|
||||||
|
} finally {
|
||||||
|
lock.writeLock().unlock();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -96,10 +101,15 @@ public class EventBus {
|
|||||||
* @param clazz the class
|
* @param clazz the class
|
||||||
* @param handler the handler
|
* @param handler the handler
|
||||||
*/
|
*/
|
||||||
public synchronized void unsubscribe(Class<?> clazz, EventHandler handler) {
|
public void unsubscribe(Class<?> clazz, EventHandler handler) {
|
||||||
checkNotNull(clazz);
|
checkNotNull(clazz);
|
||||||
checkNotNull(handler);
|
checkNotNull(handler);
|
||||||
|
lock.writeLock().lock();
|
||||||
|
try {
|
||||||
handlersByType.remove(clazz, handler);
|
handlersByType.remove(clazz, handler);
|
||||||
|
} finally {
|
||||||
|
lock.writeLock().unlock();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -107,15 +117,15 @@ public class EventBus {
|
|||||||
*
|
*
|
||||||
* @param handlers a map of handlers
|
* @param handlers a map of handlers
|
||||||
*/
|
*/
|
||||||
public synchronized void unsubscribeAll(Multimap<Class<?>, EventHandler> handlers) {
|
public void unsubscribeAll(Multimap<Class<?>, EventHandler> handlers) {
|
||||||
checkNotNull(handlers);
|
checkNotNull(handlers);
|
||||||
|
lock.writeLock().lock();
|
||||||
|
try {
|
||||||
for (Map.Entry<Class<?>, Collection<EventHandler>> entry : handlers.asMap().entrySet()) {
|
for (Map.Entry<Class<?>, Collection<EventHandler>> entry : handlers.asMap().entrySet()) {
|
||||||
Set<EventHandler> currentHandlers = getHandlersForEventType(entry.getKey());
|
handlersByType.get(entry.getKey()).removeAll(entry.getValue());
|
||||||
Collection<EventHandler> eventMethodsInListener = entry.getValue();
|
|
||||||
|
|
||||||
if (currentHandlers != null &&!currentHandlers.containsAll(entry.getValue())) {
|
|
||||||
currentHandlers.removeAll(eventMethodsInListener);
|
|
||||||
}
|
}
|
||||||
|
} finally {
|
||||||
|
lock.writeLock().unlock();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -146,25 +156,23 @@ public class EventBus {
|
|||||||
* successfully after the event has been posted to all handlers, and
|
* successfully after the event has been posted to all handlers, and
|
||||||
* regardless of any exceptions thrown by handlers.
|
* regardless of any exceptions thrown by handlers.
|
||||||
*
|
*
|
||||||
* <p>If no handlers have been subscribed for {@code event}'s class, and
|
|
||||||
* {@code event} is not already a {@link DeadEvent}, it will be wrapped in a
|
|
||||||
* DeadEvent and reposted.
|
|
||||||
*
|
|
||||||
* @param event event to post.
|
* @param event event to post.
|
||||||
*/
|
*/
|
||||||
public void post(Object event) {
|
public void post(Object event) {
|
||||||
List<EventHandler> dispatching = new ArrayList<>();
|
List<EventHandler> dispatching = new ArrayList<>();
|
||||||
|
|
||||||
synchronized (this) {
|
Set<Class<?>> dispatchTypes = flattenHierarchyCache.get(event.getClass());
|
||||||
Set<Class<?>> dispatchTypes = flattenHierarchy(event.getClass());
|
lock.readLock().lock();
|
||||||
|
try {
|
||||||
for (Class<?> eventType : dispatchTypes) {
|
for (Class<?> eventType : dispatchTypes) {
|
||||||
Set<EventHandler> wrappers = getHandlersForEventType(eventType);
|
Set<EventHandler> wrappers = handlersByType.get(eventType);
|
||||||
|
|
||||||
if (wrappers != null && !wrappers.isEmpty()) {
|
if (wrappers != null && !wrappers.isEmpty()) {
|
||||||
dispatching.addAll(wrappers);
|
dispatching.addAll(wrappers);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} finally {
|
||||||
|
lock.readLock().unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
Collections.sort(dispatching);
|
Collections.sort(dispatching);
|
||||||
@ -175,14 +183,12 @@ public class EventBus {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dispatches {@code event} to the handler in {@code handler}. This method
|
* Dispatches {@code event} to the handler in {@code handler}.
|
||||||
* is an appropriate override point for subclasses that wish to make
|
|
||||||
* event delivery asynchronous.
|
|
||||||
*
|
*
|
||||||
* @param event event to dispatch.
|
* @param event event to dispatch.
|
||||||
* @param handler handler that will call the handler.
|
* @param handler handler that will call the handler.
|
||||||
*/
|
*/
|
||||||
protected void dispatch(Object event, EventHandler handler) {
|
private void dispatch(Object event, EventHandler handler) {
|
||||||
try {
|
try {
|
||||||
handler.handleEvent(event);
|
handler.handleEvent(event);
|
||||||
} catch (InvocationTargetException e) {
|
} catch (InvocationTargetException e) {
|
||||||
@ -190,39 +196,4 @@ public class EventBus {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves a mutable set of the currently registered handlers for
|
|
||||||
* {@code type}. If no handlers are currently registered for {@code type},
|
|
||||||
* this method may either return {@code null} or an empty set.
|
|
||||||
*
|
|
||||||
* @param type type of handlers to retrieve.
|
|
||||||
* @return currently registered handlers, or {@code null}.
|
|
||||||
*/
|
|
||||||
synchronized Set<EventHandler> getHandlersForEventType(Class<?> type) {
|
|
||||||
return handlersByType.get(type);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a new Set for insertion into the handler map. This is provided
|
|
||||||
* as an override point for subclasses. The returned set should support
|
|
||||||
* concurrent access.
|
|
||||||
*
|
|
||||||
* @return a new, mutable set for handlers.
|
|
||||||
*/
|
|
||||||
protected synchronized Set<EventHandler> newHandlerSet() {
|
|
||||||
return new HashSet<>();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Flattens a class's type hierarchy into a set of Class objects. The set
|
|
||||||
* will include all superclasses (transitively), and all interfaces
|
|
||||||
* implemented by these superclasses.
|
|
||||||
*
|
|
||||||
* @param concreteClass class whose type hierarchy will be retrieved.
|
|
||||||
* @return {@code clazz}'s complete type hierarchy, flattened and uniqued.
|
|
||||||
*/
|
|
||||||
Set<Class<?>> flattenHierarchy(Class<?> concreteClass) {
|
|
||||||
return flattenHierarchyCache.get(concreteClass);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -19,34 +19,27 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.util.eventbus;
|
package com.sk89q.worldedit.util.eventbus;
|
||||||
|
|
||||||
|
import com.google.common.cache.CacheBuilder;
|
||||||
|
import com.google.common.cache.CacheLoader;
|
||||||
|
import com.google.common.cache.LoadingCache;
|
||||||
import com.google.common.collect.Lists;
|
import com.google.common.collect.Lists;
|
||||||
import com.google.common.collect.Sets;
|
import com.google.common.collect.Sets;
|
||||||
import com.sk89q.worldedit.internal.annotation.RequiresNewerGuava;
|
|
||||||
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.WeakHashMap;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Holds a cache of class hierarchy.
|
* Holds a cache of class hierarchy.
|
||||||
*
|
|
||||||
* <p>This exists because Bukkit has an ancient version of Guava and the cache
|
|
||||||
* library in Guava has since changed.</>
|
|
||||||
*/
|
*/
|
||||||
@RequiresNewerGuava
|
|
||||||
class HierarchyCache {
|
class HierarchyCache {
|
||||||
|
|
||||||
private final Map<Class<?>, Set<Class<?>>> cache = new WeakHashMap<>();
|
private final LoadingCache<Class<?>, Set<Class<?>>> cache = CacheBuilder.newBuilder()
|
||||||
|
.weakKeys()
|
||||||
|
.build(CacheLoader.from(this::build));
|
||||||
|
|
||||||
public Set<Class<?>> get(Class<?> concreteClass) {
|
public Set<Class<?>> get(Class<?> concreteClass) {
|
||||||
Set<Class<?>> ret = cache.get(concreteClass);
|
return cache.getUnchecked(concreteClass);
|
||||||
if (ret == null) {
|
|
||||||
ret = build(concreteClass);
|
|
||||||
cache.put(concreteClass, ret);
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Set<Class<?>> build(Class<?> concreteClass) {
|
protected Set<Class<?>> build(Class<?> concreteClass) {
|
||||||
|
@ -0,0 +1,53 @@
|
|||||||
|
package com.sk89q.worldedit.util.eventbus;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import static java.util.Arrays.asList;
|
||||||
|
import static java.util.Collections.singletonList;
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
public class EventBusTest {
|
||||||
|
|
||||||
|
private static final class Event {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class Subscriber {
|
||||||
|
|
||||||
|
private final List<Event> events = new ArrayList<>();
|
||||||
|
|
||||||
|
@Subscribe
|
||||||
|
public void onEvent(Event event) {
|
||||||
|
events.add(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private EventBus eventBus = new EventBus();
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRegister() {
|
||||||
|
Subscriber subscriber = new Subscriber();
|
||||||
|
eventBus.register(subscriber);
|
||||||
|
Event e1 = new Event();
|
||||||
|
eventBus.post(e1);
|
||||||
|
Event e2 = new Event();
|
||||||
|
eventBus.post(e2);
|
||||||
|
assertEquals(asList(e1, e2), subscriber.events);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUnregister() {
|
||||||
|
Subscriber subscriber = new Subscriber();
|
||||||
|
eventBus.register(subscriber);
|
||||||
|
Event e1 = new Event();
|
||||||
|
eventBus.post(e1);
|
||||||
|
eventBus.unregister(subscriber);
|
||||||
|
Event e2 = new Event();
|
||||||
|
eventBus.post(e2);
|
||||||
|
assertEquals(singletonList(e1), subscriber.events);
|
||||||
|
}
|
||||||
|
}
|
Laden…
In neuem Issue referenzieren
Einen Benutzer sperren