com.github.themrmilchmann.mjl.events.EventDispatcher Maven / Gradle / Ivy
/*
* Copyright 2018 Leon Linhart
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.github.themrmilchmann.mjl.events;
import java.util.ArrayDeque;
import java.util.Collection;
import java.util.Objects;
import java.util.Queue;
/**
* An object that takes care of submitting events to subscribers.
*
* @since 1.0.0
*
* @author Leon Linhart
*/
public abstract class EventDispatcher {
/**
* Returns a dispatcher that dispatches events directly.
*
* The dispatcher dispatches all events directly upon receiving them without any additional processing or
* computation. Thus, events are dispatched in the order in which they are received.
*
* @return a dispatcher that dispatches events directly
*
* @since 1.0.0
*/
public static EventDispatcher directDispatcher() {
return DirectDispatcher.INSTANCE;
}
/**
* Returns a dispatcher that that guarantees that all events that are posted in a single thread are dispatched to
* their subscribers in the order they are posted by queuing events that are posted reentrantly on a thread.
*
* @return a dispatcher that guarantees that all events that are posted in a single thread are dispatched to their
* subscribers in the order they are posted.
*
* @since 1.0.0
*/
@SuppressWarnings("WeakerAccess")
public static EventDispatcher perThreadDispatchQueue() {
return new PerThreadDispatchQueueDispatcher();
}
/**
* Dispatches the event to the subscribers.
*
* @param event the event to dispatch
* @param subscribers the subscribers to dispatch the event to
*
* @since 1.0.0
*/
protected abstract void dispatch(Event event, Collection subscribers);
private static class DirectDispatcher extends EventDispatcher {
private static final EventDispatcher INSTANCE = new DirectDispatcher();
@Override
protected void dispatch(Event event, Collection subscribers) {
Objects.requireNonNull(event);
Objects.requireNonNull(subscribers);
subscribers.forEach(subscriber -> subscriber.dispatch(event));
}
}
private static class PerThreadDispatchQueueDispatcher extends EventDispatcher {
private final ThreadLocal> threadLocalQueue = ThreadLocal.withInitial(ArrayDeque::new);
private final ThreadLocal isThreadDispatching = ThreadLocal.withInitial(() -> false);
@Override
protected void dispatch(Event event, Collection subscribers) {
Objects.requireNonNull(event);
Objects.requireNonNull(subscribers);
Queue eventQueue = threadLocalQueue.get();
eventQueue.offer(new QueuedEvent(event, subscribers));
if (!isThreadDispatching.get()) {
isThreadDispatching.set(true);
try {
QueuedEvent queuedEvent;
while ((queuedEvent = eventQueue.poll()) != null) {
Event e = queuedEvent.event;
queuedEvent.subscribers.forEach(subscriber -> subscriber.dispatch(e));
}
} finally {
threadLocalQueue.remove();
isThreadDispatching.remove();
}
}
}
private static class QueuedEvent {
private final Event event;
private final Collection subscribers;
private QueuedEvent(Event event, Collection subscribers) {
this.event = event;
this.subscribers = subscribers;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy