com.github.benmanes.caffeine.cache.Scheduler Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of caffeine Show documentation
Show all versions of caffeine Show documentation
A high performance caching library
/*
* Copyright 2019 Ben Manes. 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 com.github.benmanes.caffeine.cache;
import static java.util.Objects.requireNonNull;
import java.io.Serializable;
import java.lang.System.Logger;
import java.lang.System.Logger.Level;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.Future;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
/**
* A scheduler that submits a task to an executor after a given delay.
*
* @author [email protected] (Ben Manes)
*/
@FunctionalInterface
public interface Scheduler {
/**
* Returns a future that will submit the task to the given executor after the given delay.
*
* @param executor the executor to run the task
* @param command the runnable task to schedule
* @param delay how long to delay, in units of {@code unit}
* @param unit a {@code TimeUnit} determining how to interpret the {@code delay} parameter
* @return a scheduled future representing pending submission of the task
*/
Future> schedule(Executor executor, Runnable command, long delay, TimeUnit unit);
/**
* Returns a scheduler that always returns a successfully completed future.
*
* @return a scheduler that always returns a successfully completed future
*/
static Scheduler disabledScheduler() {
return DisabledScheduler.INSTANCE;
}
/**
* Returns a scheduler that uses the system-wide scheduling thread by using
* {@link CompletableFuture#delayedExecutor}.
*
* @return a scheduler that uses the system-wide scheduling thread
*/
static Scheduler systemScheduler() {
return SystemScheduler.INSTANCE;
}
/**
* Returns a scheduler that delegates to the a {@link ScheduledExecutorService}.
*
* @param scheduledExecutorService the executor to schedule on
* @return a scheduler that delegates to the a {@link ScheduledExecutorService}
*/
static Scheduler forScheduledExecutorService(ScheduledExecutorService scheduledExecutorService) {
return new ExecutorServiceScheduler(scheduledExecutorService);
}
/**
* Returns a scheduler that suppresses and logs any exception thrown by the delegate
* {@code scheduler}.
*
* @param scheduler the scheduler to delegate to
* @return an scheduler that suppresses and logs any exception thrown by the delegate
*/
static Scheduler guardedScheduler(Scheduler scheduler) {
return (scheduler instanceof GuardedScheduler) ? scheduler : new GuardedScheduler(scheduler);
}
}
enum SystemScheduler implements Scheduler {
INSTANCE;
@Override
public Future> schedule(Executor executor, Runnable command, long delay, TimeUnit unit) {
Executor delayedExecutor = CompletableFuture.delayedExecutor(delay, unit, executor);
return CompletableFuture.runAsync(command, delayedExecutor);
}
}
final class ExecutorServiceScheduler implements Scheduler, Serializable {
static final Logger logger = System.getLogger(ExecutorServiceScheduler.class.getName());
static final long serialVersionUID = 1;
final ScheduledExecutorService scheduledExecutorService;
ExecutorServiceScheduler(ScheduledExecutorService scheduledExecutorService) {
this.scheduledExecutorService = requireNonNull(scheduledExecutorService);
}
@Override
public Future> schedule(Executor executor, Runnable command, long delay, TimeUnit unit) {
requireNonNull(executor);
requireNonNull(command);
requireNonNull(unit);
if (scheduledExecutorService.isShutdown()) {
return DisabledFuture.INSTANCE;
}
return scheduledExecutorService.schedule(() -> {
try {
executor.execute(command);
} catch (Throwable t) {
logger.log(Level.WARNING, "Exception thrown when submitting scheduled task", t);
throw t;
}
}, delay, unit);
}
}
final class GuardedScheduler implements Scheduler, Serializable {
static final Logger logger = System.getLogger(GuardedScheduler.class.getName());
static final long serialVersionUID = 1;
final Scheduler delegate;
GuardedScheduler(Scheduler delegate) {
this.delegate = requireNonNull(delegate);
}
@Override
public Future> schedule(Executor executor, Runnable command, long delay, TimeUnit unit) {
try {
Future> future = delegate.schedule(executor, command, delay, unit);
return (future == null) ? DisabledFuture.INSTANCE : future;
} catch (Throwable t) {
logger.log(Level.WARNING, "Exception thrown by scheduler; discarded task", t);
return DisabledFuture.INSTANCE;
}
}
}
enum DisabledScheduler implements Scheduler {
INSTANCE;
@Override
public Future schedule(Executor executor, Runnable command, long delay, TimeUnit unit) {
requireNonNull(executor);
requireNonNull(command);
requireNonNull(unit);
return DisabledFuture.INSTANCE;
}
}
@SuppressWarnings("CheckedExceptionNotThrown")
enum DisabledFuture implements Future {
INSTANCE;
@Override public boolean isDone() {
return true;
}
@Override public boolean isCancelled() {
return false;
}
@Override public boolean cancel(boolean mayInterruptIfRunning) {
return false;
}
@Override public Void get() throws InterruptedException, ExecutionException {
return null;
}
@Override public Void get(long timeout, TimeUnit unit)
throws InterruptedException, ExecutionException, TimeoutException {
requireNonNull(unit);
return null;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy