diff --git a/EventBus/src/org/greenrobot/eventbus/AsyncPoster.java b/EventBus/src/org/greenrobot/eventbus/AsyncPoster.java index 90a30d1e..ded481ce 100644 --- a/EventBus/src/org/greenrobot/eventbus/AsyncPoster.java +++ b/EventBus/src/org/greenrobot/eventbus/AsyncPoster.java @@ -31,7 +31,7 @@ class AsyncPoster implements Runnable, Poster { queue = new PendingPostQueue(); } - public void enqueue(Subscription subscription, Object event) { + public void enqueue(ISubscription subscription, Object event) { PendingPost pendingPost = PendingPost.obtainPendingPost(subscription, event); queue.enqueue(pendingPost); eventBus.getExecutorService().execute(this); diff --git a/EventBus/src/org/greenrobot/eventbus/BackgroundPoster.java b/EventBus/src/org/greenrobot/eventbus/BackgroundPoster.java index 624ddf6d..4467ff49 100644 --- a/EventBus/src/org/greenrobot/eventbus/BackgroundPoster.java +++ b/EventBus/src/org/greenrobot/eventbus/BackgroundPoster.java @@ -34,7 +34,7 @@ final class BackgroundPoster implements Runnable, Poster { queue = new PendingPostQueue(); } - public void enqueue(Subscription subscription, Object event) { + public void enqueue(ISubscription subscription, Object event) { PendingPost pendingPost = PendingPost.obtainPendingPost(subscription, event); synchronized (this) { queue.enqueue(pendingPost); diff --git a/EventBus/src/org/greenrobot/eventbus/EventBus.java b/EventBus/src/org/greenrobot/eventbus/EventBus.java index aa6c5981..6f4b42f0 100644 --- a/EventBus/src/org/greenrobot/eventbus/EventBus.java +++ b/EventBus/src/org/greenrobot/eventbus/EventBus.java @@ -18,9 +18,12 @@ import java.lang.reflect.InvocationTargetException; import java.util.ArrayList; import java.util.HashMap; +import java.util.Iterator; +import java.util.LinkedList; import java.util.List; import java.util.Map; import java.util.Set; +import java.util.WeakHashMap; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.CopyOnWriteArrayList; import java.util.concurrent.ExecutorService; @@ -47,7 +50,7 @@ public class EventBus { private static final EventBusBuilder DEFAULT_BUILDER = new EventBusBuilder(); private static final Map, List>> eventTypesCache = new HashMap<>(); - private final Map, CopyOnWriteArrayList> subscriptionsByEventType; + private final Map, CopyOnWriteArrayList> subscriptionsByEventType; private final Map>> typesBySubscriber; private final Map, Object> stickyEvents; @@ -112,7 +115,7 @@ public EventBus() { EventBus(EventBusBuilder builder) { logger = builder.getLogger(); subscriptionsByEventType = new HashMap<>(); - typesBySubscriber = new HashMap<>(); + typesBySubscriber = new WeakHashMap<>(); stickyEvents = new ConcurrentHashMap<>(); mainThreadSupport = builder.getMainThreadSupport(); mainThreadPoster = mainThreadSupport != null ? mainThreadSupport.createPoster(this) : null; @@ -130,6 +133,24 @@ public EventBus() { executorService = builder.executorService; } + /** + * Registers the given subscriber to receive events. Like register() but doesn't maintain a reference of the subscriber. + * Subscribers don't have to call {@link #unregister(Object)} even when they are no longer interested in receiving events. + *

+ * Subscribers have event handling methods that must be annotated by {@link Subscribe}. + * The {@link Subscribe} annotation also allows configuration like {@link + * ThreadMode} and priority. + */ + public void registerWeak(Object subscriber) { + Class subscriberClass = subscriber.getClass(); + List subscriberMethods = subscriberMethodFinder.findSubscriberMethods(subscriberClass); + synchronized (this) { + for (SubscriberMethod subscriberMethod : subscriberMethods) { + subscribe(subscriber, subscriberMethod, new WeakSubscription(subscriber, subscriberMethod)); + } + } + } + /** * Registers the given subscriber to receive events. Subscribers must call {@link #unregister(Object)} once they * are no longer interested in receiving events. @@ -143,16 +164,15 @@ public void register(Object subscriber) { List subscriberMethods = subscriberMethodFinder.findSubscriberMethods(subscriberClass); synchronized (this) { for (SubscriberMethod subscriberMethod : subscriberMethods) { - subscribe(subscriber, subscriberMethod); + subscribe(subscriber, subscriberMethod, new Subscription(subscriber, subscriberMethod)); } } } // Must be called in synchronized block - private void subscribe(Object subscriber, SubscriberMethod subscriberMethod) { + private void subscribe(Object subscriber, SubscriberMethod subscriberMethod, ISubscription newSubscription) { Class eventType = subscriberMethod.eventType; - Subscription newSubscription = new Subscription(subscriber, subscriberMethod); - CopyOnWriteArrayList subscriptions = subscriptionsByEventType.get(eventType); + CopyOnWriteArrayList subscriptions = subscriptionsByEventType.get(eventType); if (subscriptions == null) { subscriptions = new CopyOnWriteArrayList<>(); subscriptionsByEventType.put(eventType, subscriptions); @@ -161,11 +181,16 @@ private void subscribe(Object subscriber, SubscriberMethod subscriberMethod) { throw new EventBusException("Subscriber " + subscriber.getClass() + " already registered to event " + eventType); } + CopyOnWriteArrayList returnedSubscriptions = pruneEmptySubscription(subscriptions); + if (returnedSubscriptions != subscriptions) { + subscriptions = returnedSubscriptions; + subscriptionsByEventType.put(eventType, returnedSubscriptions); + } } int size = subscriptions.size(); for (int i = 0; i <= size; i++) { - if (i == size || subscriberMethod.priority > subscriptions.get(i).subscriberMethod.priority) { + if (i == size || subscriberMethod.priority > subscriptions.get(i).getSubscriberMethod().priority) { subscriptions.add(i, newSubscription); break; } @@ -199,7 +224,7 @@ private void subscribe(Object subscriber, SubscriberMethod subscriberMethod) { } } - private void checkPostStickyEventToSubscription(Subscription newSubscription, Object stickyEvent) { + private void checkPostStickyEventToSubscription(ISubscription newSubscription, Object stickyEvent) { if (stickyEvent != null) { // If the subscriber is trying to abort the event, it will fail (event is not tracked in posting state) // --> Strange corner case, which we don't take care of here. @@ -223,13 +248,13 @@ public synchronized boolean isRegistered(Object subscriber) { /** Only updates subscriptionsByEventType, not typesBySubscriber! Caller must update typesBySubscriber. */ private void unsubscribeByEventType(Object subscriber, Class eventType) { - List subscriptions = subscriptionsByEventType.get(eventType); + List subscriptions = subscriptionsByEventType.get(eventType); if (subscriptions != null) { int size = subscriptions.size(); for (int i = 0; i < size; i++) { - Subscription subscription = subscriptions.get(i); - if (subscription.subscriber == subscriber) { - subscription.active = false; + ISubscription subscription = subscriptions.get(i); + if (subscription.getSubscriber() == subscriber) { + subscription.setIsActive(false); subscriptions.remove(i); i--; size--; @@ -290,7 +315,7 @@ public void cancelEventDelivery(Object event) { throw new EventBusException("Event may not be null"); } else if (postingState.event != event) { throw new EventBusException("Only the currently handled event may be aborted"); - } else if (postingState.subscription.subscriberMethod.threadMode != ThreadMode.POSTING) { + } else if (postingState.subscription.getSubscriberMethod().threadMode != ThreadMode.POSTING) { throw new EventBusException(" event handlers may only abort the incoming event"); } @@ -364,7 +389,7 @@ public boolean hasSubscriberForEvent(Class eventClass) { int countTypes = eventTypes.size(); for (int h = 0; h < countTypes; h++) { Class clazz = eventTypes.get(h); - CopyOnWriteArrayList subscriptions; + CopyOnWriteArrayList subscriptions; synchronized (this) { subscriptions = subscriptionsByEventType.get(clazz); } @@ -401,12 +426,12 @@ private void postSingleEvent(Object event, PostingThreadState postingState) thro } private boolean postSingleEventForEventType(Object event, PostingThreadState postingState, Class eventClass) { - CopyOnWriteArrayList subscriptions; + CopyOnWriteArrayList subscriptions; synchronized (this) { subscriptions = subscriptionsByEventType.get(eventClass); } if (subscriptions != null && !subscriptions.isEmpty()) { - for (Subscription subscription : subscriptions) { + for (ISubscription subscription : subscriptions) { postingState.event = event; postingState.subscription = subscription; boolean aborted; @@ -427,8 +452,26 @@ private boolean postSingleEventForEventType(Object event, PostingThreadState pos return false; } - private void postToSubscription(Subscription subscription, Object event, boolean isMainThread) { - switch (subscription.subscriberMethod.threadMode) { + private CopyOnWriteArrayList pruneEmptySubscription(CopyOnWriteArrayList origin) { + boolean modified = false; + LinkedList copy = new LinkedList<>(origin); + Iterator iterator = copy.iterator(); + while (iterator.hasNext()) { + ISubscription subscription = iterator.next(); + if (subscription.getSubscriber() == null) { + iterator.remove(); + modified = true; + } + } + if (modified) { + return new CopyOnWriteArrayList<>(copy); + } else { + return origin; + } + } + + private void postToSubscription(ISubscription subscription, Object event, boolean isMainThread) { + switch (subscription.getSubscriberMethod().threadMode) { case POSTING: invokeSubscriber(subscription, event); break; @@ -458,7 +501,7 @@ private void postToSubscription(Subscription subscription, Object event, boolean asyncPoster.enqueue(subscription, event); break; default: - throw new IllegalStateException("Unknown thread mode: " + subscription.subscriberMethod.threadMode); + throw new IllegalStateException("Unknown thread mode: " + subscription.getSubscriberMethod().threadMode); } } @@ -498,28 +541,32 @@ static void addInterfaces(List> eventTypes, Class[] interfaces) { */ void invokeSubscriber(PendingPost pendingPost) { Object event = pendingPost.event; - Subscription subscription = pendingPost.subscription; + ISubscription subscription = pendingPost.subscription; PendingPost.releasePendingPost(pendingPost); - if (subscription.active) { + if (subscription.isActive()) { invokeSubscriber(subscription, event); } } - void invokeSubscriber(Subscription subscription, Object event) { - try { - subscription.subscriberMethod.method.invoke(subscription.subscriber, event); - } catch (InvocationTargetException e) { - handleSubscriberException(subscription, event, e.getCause()); - } catch (IllegalAccessException e) { - throw new IllegalStateException("Unexpected exception", e); + void invokeSubscriber(ISubscription subscription, Object event) { + Object subscriber = subscription.getSubscriber(); + if (subscriber != null) { + try { + subscription.getSubscriberMethod().method.invoke(subscriber, event); + } catch (InvocationTargetException e) { + handleSubscriberException(subscription, event, e.getCause()); + } catch (IllegalAccessException e) { + throw new IllegalStateException("Unexpected exception", e); + } } } - private void handleSubscriberException(Subscription subscription, Object event, Throwable cause) { + private void handleSubscriberException(ISubscription subscription, Object event, Throwable cause) { + Object subscriber = subscription.getSubscriber(); if (event instanceof SubscriberExceptionEvent) { if (logSubscriberExceptions) { // Don't send another SubscriberExceptionEvent to avoid infinite event recursion, just log - logger.log(Level.SEVERE, "SubscriberExceptionEvent subscriber " + subscription.subscriber.getClass() + logger.log(Level.SEVERE, "SubscriberExceptionEvent subscriber " + (subscriber != null ? subscriber.getClass() : null) + " threw an exception", cause); SubscriberExceptionEvent exEvent = (SubscriberExceptionEvent) event; logger.log(Level.SEVERE, "Initial event " + exEvent.causingEvent + " caused exception in " @@ -531,11 +578,11 @@ private void handleSubscriberException(Subscription subscription, Object event, } if (logSubscriberExceptions) { logger.log(Level.SEVERE, "Could not dispatch event: " + event.getClass() + " to subscribing class " - + subscription.subscriber.getClass(), cause); + + (subscriber != null ? subscriber.getClass() : null), cause); } if (sendSubscriberExceptionEvent) { SubscriberExceptionEvent exEvent = new SubscriberExceptionEvent(this, cause, event, - subscription.subscriber); + subscription.getSubscriber()); post(exEvent); } } @@ -546,7 +593,7 @@ final static class PostingThreadState { final List eventQueue = new ArrayList<>(); boolean isPosting; boolean isMainThread; - Subscription subscription; + ISubscription subscription; Object event; boolean canceled; } diff --git a/EventBus/src/org/greenrobot/eventbus/HandlerPoster.java b/EventBus/src/org/greenrobot/eventbus/HandlerPoster.java index 95309547..308add02 100644 --- a/EventBus/src/org/greenrobot/eventbus/HandlerPoster.java +++ b/EventBus/src/org/greenrobot/eventbus/HandlerPoster.java @@ -34,7 +34,7 @@ protected HandlerPoster(EventBus eventBus, Looper looper, int maxMillisInsideHan queue = new PendingPostQueue(); } - public void enqueue(Subscription subscription, Object event) { + public void enqueue(ISubscription subscription, Object event) { PendingPost pendingPost = PendingPost.obtainPendingPost(subscription, event); synchronized (this) { queue.enqueue(pendingPost); diff --git a/EventBus/src/org/greenrobot/eventbus/ISubscription.java b/EventBus/src/org/greenrobot/eventbus/ISubscription.java new file mode 100644 index 00000000..b1e71870 --- /dev/null +++ b/EventBus/src/org/greenrobot/eventbus/ISubscription.java @@ -0,0 +1,12 @@ +package org.greenrobot.eventbus; + +public interface ISubscription { + + public Object getSubscriber(); + + public SubscriberMethod getSubscriberMethod(); + + public boolean isActive(); + + public void setIsActive(boolean isActive); +} diff --git a/EventBus/src/org/greenrobot/eventbus/PendingPost.java b/EventBus/src/org/greenrobot/eventbus/PendingPost.java index 01f474c2..58e85d83 100644 --- a/EventBus/src/org/greenrobot/eventbus/PendingPost.java +++ b/EventBus/src/org/greenrobot/eventbus/PendingPost.java @@ -22,15 +22,15 @@ final class PendingPost { private final static List pendingPostPool = new ArrayList(); Object event; - Subscription subscription; + ISubscription subscription; PendingPost next; - private PendingPost(Object event, Subscription subscription) { + private PendingPost(Object event, ISubscription subscription) { this.event = event; this.subscription = subscription; } - static PendingPost obtainPendingPost(Subscription subscription, Object event) { + static PendingPost obtainPendingPost(ISubscription subscription, Object event) { synchronized (pendingPostPool) { int size = pendingPostPool.size(); if (size > 0) { diff --git a/EventBus/src/org/greenrobot/eventbus/Poster.java b/EventBus/src/org/greenrobot/eventbus/Poster.java index a69a078d..1865c75b 100644 --- a/EventBus/src/org/greenrobot/eventbus/Poster.java +++ b/EventBus/src/org/greenrobot/eventbus/Poster.java @@ -28,5 +28,5 @@ interface Poster { * @param subscription Subscription which will receive the event. * @param event Event that will be posted to subscribers. */ - void enqueue(Subscription subscription, Object event); + void enqueue(ISubscription subscription, Object event); } diff --git a/EventBus/src/org/greenrobot/eventbus/Subscription.java b/EventBus/src/org/greenrobot/eventbus/Subscription.java index cc0de1e3..f4cef5c2 100644 --- a/EventBus/src/org/greenrobot/eventbus/Subscription.java +++ b/EventBus/src/org/greenrobot/eventbus/Subscription.java @@ -15,14 +15,34 @@ */ package org.greenrobot.eventbus; -final class Subscription { - final Object subscriber; - final SubscriberMethod subscriberMethod; +final class Subscription implements ISubscription { + private final Object subscriber; + private final SubscriberMethod subscriberMethod; /** * Becomes false as soon as {@link EventBus#unregister(Object)} is called, which is checked by queued event delivery * {@link EventBus#invokeSubscriber(PendingPost)} to prevent race conditions. */ - volatile boolean active; + private volatile boolean active; + + @Override + public Object getSubscriber() { + return subscriber; + } + + @Override + public SubscriberMethod getSubscriberMethod() { + return subscriberMethod; + } + + @Override + public boolean isActive() { + return active; + } + + @Override + public void setIsActive(boolean isActive) { + this.active = isActive; + } Subscription(Object subscriber, SubscriberMethod subscriberMethod) { this.subscriber = subscriber; @@ -32,10 +52,10 @@ final class Subscription { @Override public boolean equals(Object other) { - if (other instanceof Subscription) { - Subscription otherSubscription = (Subscription) other; - return subscriber == otherSubscription.subscriber - && subscriberMethod.equals(otherSubscription.subscriberMethod); + if (other instanceof ISubscription) { + ISubscription otherSubscription = (ISubscription) other; + return subscriber == otherSubscription.getSubscriber() + && subscriberMethod.equals(otherSubscription.getSubscriberMethod()); } else { return false; } diff --git a/EventBus/src/org/greenrobot/eventbus/WeakSubscription.java b/EventBus/src/org/greenrobot/eventbus/WeakSubscription.java new file mode 100644 index 00000000..9ca8f658 --- /dev/null +++ b/EventBus/src/org/greenrobot/eventbus/WeakSubscription.java @@ -0,0 +1,64 @@ +package org.greenrobot.eventbus; + +import java.lang.ref.WeakReference; + +public class WeakSubscription extends WeakReference implements ISubscription { + + private final SubscriberMethod subscriberMethod; + + /** + * Becomes false as soon as {@link EventBus#unregister(Object)} is called, which is checked by queued event delivery + * {@link EventBus#invokeSubscriber(PendingPost)} to prevent race conditions. + */ + private volatile boolean active; + + WeakSubscription(Object subscriber, SubscriberMethod subscriberMethod) { + super(subscriber); + this.subscriberMethod = subscriberMethod; + active = true; + } + + @Override + public boolean equals(Object other) { + if (other instanceof ISubscription) { + ISubscription otherSubscription = (ISubscription) other; + return get() == otherSubscription.getSubscriber() + && subscriberMethod.equals(otherSubscription.getSubscriberMethod()); + } else { + return false; + } + } + + @Override + public int hashCode() { + // I have to admit that this kind of implementation could not guarantee thread safe. + // If the reference is being claimed when hashCode() executing, and the equals() + // might not behave in accordance with it. But it seems there is no better way. + Object referent = get(); + if (referent != null) { + return referent.hashCode() + subscriberMethod.methodString.hashCode(); + } else { + return System.identityHashCode(this) + subscriberMethod.methodString.hashCode(); + } + } + + @Override + public Object getSubscriber() { + return get(); + } + + @Override + public SubscriberMethod getSubscriberMethod() { + return subscriberMethod; + } + + @Override + public boolean isActive() { + return active; + } + + @Override + public void setIsActive(boolean isActive) { + this.active = isActive; + } +}