Während die blockierende Warteschlange, wie von anderen vorgeschlagen, funktionieren würde, bin ich kein großer Fan davon, weil es ein beschäftigtes Warten erfordert (der Verbraucher schleift unendlich nach einer Nachricht). Eine Alternative wäre, eine Aufgabe jedes Mal zu senden, wenn die Observer
eine Benachrichtigung erhält.
public class Main extends Observable implements Observer {
private final int numCores = Runtime.getRuntime().availableProcessors();
private final ThreadPoolExecutor executor = (ThreadPoolExecutor) Executors.newFixedThreadPool(numCores);
public Main() {
addObserver(this);
}
public static void main(String[] args) throws InterruptedException {
new Main().execute();
}
private void execute() {
for (int i = 0; i < 5; ++i) {
this.setChanged();
this.notifyObservers(i);
try {
Thread.sleep(1000l);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
executor.shutdown();
}
@Override
public void update(Observable o, Object arg) {
System.out.printf("Received notification on thread: %s.\n", Thread.currentThread().getName());
executor.submit(() -> System.out.printf("Running in thread: %s, result: %s.\n",
Thread.currentThread().getName(), arg));
}
}
Ausgang:
Received notification on thread: main.
Running in thread: pool-1-thread-1, result: 0.
Received notification on thread: main.
Running in thread: pool-1-thread-2, result: 1.
Received notification on thread: main.
Running in thread: pool-1-thread-3, result: 2.
Received notification on thread: main.
Running in thread: pool-1-thread-4, result: 3.
Received notification on thread: main.
Running in thread: pool-1-thread-5, result: 4.
ein Blocking verwenden. Senden Sie das Ereignis in der Warteschlange, wenn update() aufgerufen wird. Lies das Ereignis von der BlockingQueue in run(). –