
reactor.core.scheduler.ParallelScheduler Maven / Gradle / Ivy
/*
* Copyright (c) 2011-2017 Pivotal Software Inc, All Rights Reserved.
*
* 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 reactor.core.scheduler;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
import java.util.function.Supplier;
import reactor.core.Disposable;
/**
* Scheduler that hosts a fixed pool of single-threaded ScheduledExecutorService-based workers
* and is suited for parallel work. This scheduler is time-capable (can schedule with
* delay / periodically).
*
* @author Stephane Maldini
* @author Simon Baslé
*/
final class ParallelScheduler implements Scheduler, Supplier {
static final AtomicLong COUNTER = new AtomicLong();
final int n;
final ThreadFactory factory;
volatile ScheduledExecutorService[] executors;
static final AtomicReferenceFieldUpdater EXECUTORS =
AtomicReferenceFieldUpdater.newUpdater(ParallelScheduler.class, ScheduledExecutorService[].class, "executors");
static final ScheduledExecutorService[] SHUTDOWN = new ScheduledExecutorService[0];
static final ScheduledExecutorService TERMINATED;
static {
TERMINATED = Executors.newSingleThreadScheduledExecutor();
TERMINATED.shutdownNow();
}
int roundRobin;
ParallelScheduler(int n, ThreadFactory factory) {
if (n <= 0) {
throw new IllegalArgumentException("n > 0 required but it was " + n);
}
this.n = n;
this.factory = factory;
init(n);
}
/**
* Instantiates the default {@link ScheduledExecutorService} for the ParallelScheduler
* ({@code Executors.newSingleThreadScheduledExecutor}).
*/
@Override
public ScheduledExecutorService get() {
return Executors.newSingleThreadScheduledExecutor(factory);
}
void init(int n) {
ScheduledExecutorService[] a = new ScheduledExecutorService[n];
for (int i = 0; i < n; i++) {
a[i] = Schedulers.decorateExecutorService(Schedulers.PARALLEL, this);
}
EXECUTORS.lazySet(this, a);
}
@Override
public boolean isDisposed() {
return executors == SHUTDOWN;
}
@Override
public void start() {
ScheduledExecutorService[] b = null;
for (;;) {
ScheduledExecutorService[] a = executors;
if (a != SHUTDOWN) {
if (b != null) {
for (ScheduledExecutorService exec : b) {
exec.shutdownNow();
}
}
return;
}
if (b == null) {
b = new ScheduledExecutorService[n];
for (int i = 0; i < n; i++) {
b[i] = Schedulers.decorateExecutorService(Schedulers.PARALLEL, this);
}
}
if (EXECUTORS.compareAndSet(this, a, b)) {
return;
}
}
}
@Override
public void dispose() {
ScheduledExecutorService[] a = executors;
if (a != SHUTDOWN) {
a = EXECUTORS.getAndSet(this, SHUTDOWN);
if (a != SHUTDOWN) {
for (ScheduledExecutorService exec : a) {
exec.shutdownNow();
}
}
}
}
ScheduledExecutorService pick() {
ScheduledExecutorService[] a = executors;
if (a != SHUTDOWN) {
// ignoring the race condition here, its already random who gets which executor
int idx = roundRobin;
if (idx == n) {
idx = 0;
roundRobin = 1;
} else {
roundRobin = idx + 1;
}
return a[idx];
}
return TERMINATED;
}
@Override
public Disposable schedule(Runnable task) {
return Schedulers.directSchedule(pick(), task, 0L, TimeUnit.MILLISECONDS);
}
@Override
public Disposable schedule(Runnable task, long delay, TimeUnit unit) {
return Schedulers.directSchedule(pick(), task, delay, unit);
}
@Override
public Disposable schedulePeriodically(Runnable task,
long initialDelay,
long period,
TimeUnit unit) {
return Schedulers.directSchedulePeriodically(pick(),
task,
initialDelay,
period,
unit);
}
@Override
public Worker createWorker() {
return new ExecutorServiceWorker(pick());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy