monix.execution.schedulers.SchedulerCompanionImpl.scala Maven / Gradle / Ivy
The newest version!
/*
* Copyright (c) 2014-2021 by The Monix Project Developers.
* See the project homepage at: https://monix.io
*
* 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 monix.execution.schedulers
import java.util.concurrent._
import monix.execution.{Features, Scheduler, SchedulerCompanion, UncaughtExceptionReporter}
// Prevents conflict with the deprecated symbol
import monix.execution.{ExecutionModel => ExecModel}
import scala.concurrent.ExecutionContext
import scala.concurrent.duration._
/** @define applyDesc The resulting [[Scheduler]] will piggyback on top of a Java
* `ScheduledExecutorService` for scheduling tasks for execution with
* a delay and a Scala `ExecutionContext` for actually executing the
* tasks.
*
* @define executorService is the `ScheduledExecutorService` that handles the
* scheduling of tasks into the future. If not provided, an internal
* default will be used. You can also create one with
* `java.util.concurrent.Executors`.
*
* @define executionContext is the execution context on top of which all tasks will run.
* Use `scala.concurrent.ExecutionContext.Implicits.global`
* for the default.
*
* @define executionModel is the preferred
* [[monix.execution.ExecutionModel ExecutionModel]],
* a guideline for run-loops and producers of data. Use
* [[monix.execution.ExecutionModel.Default ExecutionModel.Default]]
* for the default.
*
* @define reporter is the [[UncaughtExceptionReporter]] that logs uncaught exceptions.
* Wrap the given `ExecutionContext.reportFailure` or use
* [[UncaughtExceptionReporter.default]] for the default.
*/
private[execution] class SchedulerCompanionImpl extends SchedulerCompanion {
/** The default `ScheduledExecutor` instance.
*
* Currently it's a single-threaded Java `ScheduledExecutorService`
* used for scheduling delayed tasks for execution. But the actual
* execution will not happen on this executor service. In general
* you can just reuse this one for all your scheduling needs.
*/
lazy val DefaultScheduledExecutor: ScheduledExecutorService =
Defaults.scheduledExecutor
/** [[monix.execution.Scheduler Scheduler]] builder.
*
* The resulting [[Scheduler]] will piggyback on top of a Java
* `ScheduledExecutorService` for scheduling tasks for execution with
* a delay and a Scala `ExecutionContext` for actually executing the
* tasks.
*
* @param executor $executorService
* @param ec $executionContext
* @param reporter $reporter
* @param executionModel $executionModel
*/
def apply(
executor: ScheduledExecutorService,
ec: ExecutionContext,
reporter: UncaughtExceptionReporter,
executionModel: ExecModel): Scheduler =
AsyncScheduler(executor, ec, executionModel, reporter)
/** [[monix.execution.Scheduler Scheduler]] builder.
*
* @param executor $executorService
* @param ec $executionContext
*/
def apply(executor: ScheduledExecutorService, ec: ExecutionContext): Scheduler =
AsyncScheduler(executor, ec, ExecModel.Default)
/** [[monix.execution.Scheduler Scheduler]] builder.
*
* @param ec $executionContext
* @param reporter $reporter
*/
def apply(ec: ExecutionContext, reporter: UncaughtExceptionReporter): Scheduler =
AsyncScheduler(DefaultScheduledExecutor, ec, ExecModel.Default, reporter)
/** [[monix.execution.Scheduler Scheduler]] builder .
*
* @param ec $executionContext
* @param reporter $reporter
* @param executionModel $executionModel
*/
def apply(ec: ExecutionContext, reporter: UncaughtExceptionReporter, executionModel: ExecModel): Scheduler =
AsyncScheduler(DefaultScheduledExecutor, ec, executionModel, reporter)
/** [[monix.execution.Scheduler Scheduler]] builder that converts a
* Java `ExecutorService` into a scheduler.
*
* @param executor $executorService
* @param reporter $reporter
*/
def apply(executor: ExecutorService, reporter: UncaughtExceptionReporter): SchedulerService =
ExecutorScheduler(executor, reporter, ExecModel.Default, Features.empty)
/** [[monix.execution.Scheduler Scheduler]] builder that converts a
* Java `ExecutorService` into a scheduler.
*
* @param executor $executorService
* @param reporter $reporter
* @param executionModel $executionModel
*/
def apply(
executor: ExecutorService,
reporter: UncaughtExceptionReporter,
executionModel: ExecModel): SchedulerService = {
ExecutorScheduler(executor, reporter, executionModel, Features.empty)
}
/** [[monix.execution.Scheduler Scheduler]] builder that converts a
* Java `ExecutorService` into a scheduler.
*
* @param executor $executorService
*/
def apply(executor: ExecutorService): SchedulerService =
ExecutorScheduler(executor, UncaughtExceptionReporter.default, ExecModel.Default, Features.empty)
/** [[monix.execution.Scheduler Scheduler]] builder that converts a
* Java `ExecutorService` into a scheduler.
*
* @param executor $executorService
* @param executionModel $executionModel
*/
def apply(executor: ExecutorService, executionModel: ExecModel): SchedulerService =
ExecutorScheduler(executor, UncaughtExceptionReporter.default, executionModel, Features.empty)
/** [[monix.execution.Scheduler Scheduler]] builder - uses monix's
* default `ScheduledExecutorService` for handling the scheduling of tasks.
*
* @param ec is the execution context on top of which all tasks will run.
*/
def apply(ec: ExecutionContext): Scheduler =
AsyncScheduler(
DefaultScheduledExecutor,
ec,
ExecModel.Default
)
/** [[monix.execution.Scheduler Scheduler]] builder - uses monix's
* default `ScheduledExecutorService` for handling the scheduling of tasks.
*
* @param ec is the execution context on top of which all tasks will run.
* @param executionModel $executionModel
*/
def apply(ec: ExecutionContext, executionModel: ExecModel): Scheduler =
AsyncScheduler(
DefaultScheduledExecutor,
ec,
executionModel,
UncaughtExceptionReporter(ec.reportFailure)
)
/** [[monix.execution.Scheduler Scheduler]] builder - uses monix's
* default `ScheduledExecutorService` for handling the scheduling of tasks.
* Uses Scala's `s.c.ExecutionContext.global` for actual execution.
*
* @param reporter $reporter
* @param executionModel $executionModel
*/
def apply(reporter: UncaughtExceptionReporter, executionModel: ExecModel): Scheduler = {
val ec = ExecutionContext.Implicits.global
AsyncScheduler(DefaultScheduledExecutor, ec, executionModel, reporter)
}
/** [[monix.execution.Scheduler Scheduler]] builder - uses monix's
* default `ScheduledExecutorService` for handling the scheduling of tasks.
* Uses Scala's `s.c.ExecutionContext.global` for actual execution.
*
* @param executionModel $executionModel
*/
def apply(executionModel: ExecModel): Scheduler = {
val ec = ExecutionContext.Implicits.global
AsyncScheduler(
DefaultScheduledExecutor,
ec,
executionModel
)
}
/** Builds a [[monix.execution.schedulers.TrampolineScheduler TrampolineScheduler]].
*
* @param underlying is the [[monix.execution.Scheduler Scheduler]]
* to which the we defer to in case asynchronous or time-delayed
* execution is needed
*
* @define executionModel is the preferred
* [[monix.execution.ExecutionModel ExecutionModel]],
* a guideline for run-loops and producers of data. Use
* [[monix.execution.ExecutionModel.Default ExecutionModel.Default]]
* for the default.
*/
def trampoline(underlying: Scheduler = Implicits.global, executionModel: ExecModel = ExecModel.Default): Scheduler =
TrampolineScheduler(underlying, executionModel)
/** Creates a [[monix.execution.Scheduler Scheduler]] meant for
* computationally heavy CPU-bound tasks.
*
* Characteristics:
*
* - backed by a `ForkJoinPool` implementation, in async mode
* - uses monix's default `ScheduledExecutorService` instance for scheduling
* - DOES NOT cooperate with Scala's `BlockContext`
*
* @param name the created threads name prefix, for easy identification.
* @param parallelism is the number of threads that can run in parallel
* @param daemonic specifies whether the created threads should be daemonic
* (non-daemonic threads are blocking the JVM process on exit).
* @param reporter $reporter
* @param executionModel $executionModel
*/
def computation(
parallelism: Int = Runtime.getRuntime.availableProcessors(),
name: String = "monix-computation",
daemonic: Boolean = true,
reporter: UncaughtExceptionReporter = UncaughtExceptionReporter.default,
executionModel: ExecModel = ExecModel.Default): SchedulerService = {
ExecutorScheduler.forkJoinStatic(name, parallelism, daemonic, reporter, executionModel)
}
/** Creates a general purpose [[monix.execution.Scheduler Scheduler]]
* backed by a `ForkJoinPool`, similar to Scala's `global`.
*
* Characteristics:
*
* - backed by a `ForkJoinPool` implementation, in async mode
* - uses monix's default `ScheduledExecutorService` instance for scheduling
* - cooperates with Scala's `BlockContext`
*
* @param parallelism is the number of threads that can run in parallel
* @param maxThreads is the maximum number of threads that can be created
* @param name the created threads name prefix, for easy identification.
* @param daemonic specifies whether the created threads should be daemonic
* (non-daemonic threads are blocking the JVM process on exit).
* @param reporter $reporter
* @param executionModel $executionModel
*/
def forkJoin(
parallelism: Int,
maxThreads: Int,
name: String = "monix-forkjoin",
daemonic: Boolean = true,
reporter: UncaughtExceptionReporter = UncaughtExceptionReporter.default,
executionModel: ExecModel = ExecModel.Default): SchedulerService = {
ExecutorScheduler.forkJoinDynamic(name, parallelism, maxThreads, daemonic, reporter, executionModel)
}
/** Creates a [[monix.execution.Scheduler Scheduler]] meant for blocking I/O tasks.
*
* Characteristics:
*
* - backed by a cached `ThreadPool` executor with 60 seconds of keep-alive
* - the maximum number of threads is unbounded, as recommended for blocking I/O
* - uses monix's default `ScheduledExecutorService` instance for scheduling
* - doesn't cooperate with Scala's `BlockContext` because it is unbounded
*
* @param name the created threads name prefix, for easy identification.
* @param daemonic specifies whether the created threads should be daemonic
* (non-daemonic threads are blocking the JVM process on exit).
* @param reporter $reporter
* @param executionModel $executionModel
*/
def io(
name: String = "monix-io",
daemonic: Boolean = true,
reporter: UncaughtExceptionReporter = UncaughtExceptionReporter.default,
executionModel: ExecModel = ExecModel.Default): SchedulerService = {
val threadFactory = ThreadFactoryBuilder(name, reporter, daemonic)
val executor = new ThreadPoolExecutor(
0,
Int.MaxValue,
60,
TimeUnit.SECONDS,
new SynchronousQueue[Runnable](false),
threadFactory)
ExecutorScheduler(executor, reporter, executionModel, Features.empty)
}
/** Builds a [[Scheduler]] backed by an internal
* `java.util.concurrent.ThreadPoolExecutor`, that executes each submitted
* task using one of possibly several pooled threads.
*
* @param name the created threads name prefix, for easy identification
* @param minThreads the number of threads to keep in the pool, even
* if they are idle
* @param maxThreads the maximum number of threads to allow in the pool
* @param keepAliveTime when the number of threads is greater than
* the core, this is the maximum time that excess idle threads
* will wait for new tasks before terminating
* @param daemonic specifies whether the created threads should be daemonic
* (non-daemonic threads are blocking the JVM process on exit).
* @param reporter $reporter
* @param executionModel $executionModel
*/
def cached(
name: String,
minThreads: Int,
maxThreads: Int,
keepAliveTime: FiniteDuration = 60.seconds,
daemonic: Boolean = true,
reporter: UncaughtExceptionReporter = UncaughtExceptionReporter.default,
executionModel: ExecModel = ExecModel.Default): SchedulerService = {
require(minThreads >= 0, "minThreads >= 0")
require(maxThreads > 0, "maxThreads > 0")
require(maxThreads >= minThreads, "maxThreads >= minThreads")
require(keepAliveTime >= Duration.Zero, "keepAliveTime >= 0")
val threadFactory = ThreadFactoryBuilder(name, reporter, daemonic)
val executor = new ThreadPoolExecutor(
minThreads,
maxThreads,
keepAliveTime.toMillis,
TimeUnit.MILLISECONDS,
new SynchronousQueue[Runnable](false),
threadFactory)
ExecutorScheduler(executor, reporter, executionModel, Features.empty)
}
/** Builds a [[monix.execution.Scheduler Scheduler]] that schedules and executes tasks on its own thread.
*
* Characteristics:
*
* - backed by a single-threaded `ScheduledExecutorService` that takes care
* of both scheduling tasks in the future and of executing tasks
* - does not cooperate with Scala's `BlockingContext`, so tasks should not
* block on the result of other tasks scheduled to run on this same thread
*
* @param name is the name of the created thread, for easy identification
* @param daemonic specifies whether the created thread should be daemonic
* (non-daemonic threads are blocking the JVM process on exit)
* @param reporter $reporter
* @param executionModel $executionModel
*/
def singleThread(
name: String,
daemonic: Boolean = true,
reporter: UncaughtExceptionReporter = UncaughtExceptionReporter.default,
executionModel: ExecModel = ExecModel.Default): SchedulerService = {
val factory = ThreadFactoryBuilder(name, reporter, daemonic)
val executor = new AdaptedThreadPoolExecutor(1, factory) {
override def reportFailure(t: Throwable): Unit =
reporter.reportFailure(t)
}
ExecutorScheduler(executor, null, executionModel, Features.empty)
}
/** Builds a [[monix.execution.Scheduler Scheduler]] with a fixed thread-pool.
*
* Characteristics:
*
* - backed by a fixed pool `ScheduledExecutorService` that takes care
* of both scheduling tasks in the future and of executing immediate tasks
* - does not cooperate with Scala's `BlockingContext`, so tasks should not
* block on the result of other tasks scheduled to run on this same thread
*
* @param name the created threads name prefix, for easy identification.
* @param daemonic specifies whether the created thread should be daemonic
* @param reporter is the [[UncaughtExceptionReporter]] that logs uncaught exceptions.
*/
def fixedPool(
name: String,
poolSize: Int,
daemonic: Boolean = true,
reporter: UncaughtExceptionReporter = UncaughtExceptionReporter.default,
executionModel: ExecModel = ExecModel.Default): SchedulerService = {
val factory = ThreadFactoryBuilder(name, reporter, daemonic)
val executor = new AdaptedThreadPoolExecutor(poolSize, factory) {
override def reportFailure(t: Throwable): Unit =
reporter.reportFailure(t)
}
ExecutorScheduler(executor, null, executionModel, Features.empty)
}
/** The explicit global `Scheduler`. Invoke `global` when you want
* to provide the global `Scheduler` explicitly.
*
* The default `Scheduler` implementation is backed by a
* work-stealing thread pool, along with a single-threaded
* `ScheduledExecutionContext` that does the scheduling. By default,
* the thread pool uses a target number of worker threads equal
* to the number of
* [[https://docs.oracle.com/javase/8/docs/api/java/lang/Runtime.html#availableProcessors-- available processors]].
*
* @return the global `Scheduler`
*/
def global: Scheduler = Implicits.global
/** A global [[monix.execution.Scheduler Scheduler]] instance that does
* propagation of [[monix.execution.misc.Local.Context Local.Context]]
* on async execution.
*
* It wraps [[global]].
*/
def traced: Scheduler = Implicits.traced
/**
* @define globalTuning
* It can be tuned by setting the following JVM system properties:
*
* - "scala.concurrent.context.minThreads" an integer specifying the minimum
* number of active threads in the pool
*
* - "scala.concurrent.context.maxThreads" an integer specifying the maximum
* number of active threads in the pool
*
* - "scala.concurrent.context.numThreads" can be either an integer,
* specifying the parallelism directly or a string with the format "xNUM"
* (e.g. "x1.5") specifying the multiplication factor of the number of
* available processors (taken with `Runtime.availableProcessors`)
*
* The formula for calculating the parallelism in our pool is
* `min(maxThreads, max(minThreads, numThreads))`.
*
* To set a system property from the command line, a JVM parameter must be
* given to the `java` command as `-Dname=value`. So as an example, to customize
* this global scheduler, we could start our process like this:
*
*
* java -Dscala.concurrent.context.minThreads=2 \
* -Dscala.concurrent.context.maxThreads=30 \
* -Dscala.concurrent.context.numThreads=x1.5 \
* ...
*
*
* As a note, this being backed by Scala's own global execution context,
* it is cooperating with Scala's BlockContext, so when operations marked
* with `scala.concurrent.blocking` are encountered, the thread-pool may
* decide to add extra threads in the pool. However this is not a thread-pool
* that is optimal for doing blocking operations, so for example if you want
* to do a lot of blocking I/O, then use a Scheduler backed by a
* thread-pool that is more optimal for blocking. See for example
* [[io]].
*/
object Implicits extends ImplicitsLike {
/** A global [[monix.execution.Scheduler Scheduler]] instance,
* provided for convenience, piggy-backing on top of Scala's
* own `concurrent.ExecutionContext.global`, which is a
* `ForkJoinPool`.
*
* $globalTuning
*/
implicit lazy val global: Scheduler =
AsyncScheduler(
DefaultScheduledExecutor,
ExecutionContext.Implicits.global,
ExecModel.Default
)
/** A [[monix.execution.Scheduler Scheduler]] instance that does
* propagation of [[monix.execution.misc.Local.Context Local.Context]]
* through async execution.
*
* It wraps [[global]].
*
* $globalTuning
*/
implicit lazy val traced: Scheduler =
TracingScheduler(global)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy