What is the default scheduler in Project Reactor?
Asked Answered
H

2

4

Is there a "default" scheduler in the Project Reactor? Which one is it? By "default" I mean the one used when no subscribeOn() nor publishOn() are called for the chain.

Hemelytron answered 19/4, 2022 at 12:52 Comment(0)
R
5

By default, data production process starts on the Thread that initiated the subscription. Operators that deal with a time (e.g. Mono.delay) will default to running on the Schedulers.parallel() scheduler.

Most of reactive libraries (Reactive Redis, Mongo, ...) would use parallel as a default scheduler.

For example Spring WebFlux, typically use Reactor Netty as a default embedded server and would initiate subscription on Schedulers.parallel().

Rafe answered 19/4, 2022 at 13:59 Comment(4)
What about r2dbc libraries? If we subscribe to Mono/Flux that originates from r2dbc...Hemelytron
all Spring Reactive libs would use parallelRafe
any documentation that backs this up?Laszlo
@Laszlo See my detailed answer with sources: https://mcmap.net/q/663046/-what-is-the-default-scheduler-in-project-reactorSeessel
S
3

Summary

There isn't anything like a "default scheduler" in project Reactor. It depends on many factors and unless you carefully specify the scheduler, you should not make your code logic dependent on any assumptions.

Warning

As many of us use the Project Reactor library together with Spring WebFlux framework, we tend to mentally mix these two. To answer the question properly, it's good to be aware of the borderline.

Detailed answer with sources

(My answer gives more or less the same as @Alex's, but I tried to have it based on some authoritative sources.)

The Reactor documentation says that

Reactor... can be considered to be concurrency-agnostic. That is, it does not enforce a concurrency model. Rather, it leaves you, the developer, in command. However, that does not prevent the library from helping you with concurrency.

Obtaining a Flux or a Mono does not necessarily mean that it runs in a dedicated Thread. Instead, most operators continue working in the Thread on which the previous operator executed.
Unless specified, the topmost operator (the source) itself runs on the Thread in which the subscribe() call was made.

So without any meddling and with using just the "basic" operators like map(), flatMap() or filter(),
the final subscribe() runs on the calling thread.

Some operators use a specific scheduler from Schedulers by default (and usually give you the option of providing a different one). For instance, calling the Flux.interval(Duration.ofMillis(300)) factory method produces a Flux<Long> that ticks every 300ms. By default, this is enabled by Schedulers.parallel().

So just by calling some operators the default thread is changed. It is always mentioned in the documentation for the method. E.g. for Flux.interval(Duration):

Runs on the Schedulers.parallel() Scheduler.

But you may change it if you wish:

The following line changes the Scheduler to a new instance similar to Schedulers.single():

Flux.interval(Duration.ofMillis(300), Schedulers.newSingle("test"))

It seems like each Mono and Flux method which runs on a different scheduler has an overloaded alternative with an extra Scheduler parameter.

Spring WebFlux documentation tells this:

What threads should you expect to see on a server running with Spring WebFlux?

On a “vanilla” Spring WebFlux server (for example, no data access or other optional dependencies), you can expect one thread for the server and several others for request processing (typically as many as the number of CPU cores). Servlet containers, however, may start with more threads (for example, 10 on Tomcat), in support of both servlet (blocking) I/O and servlet 3.1 (non-blocking) I/O usage.

The reactive WebClient operates in event loop style. So you can see a small, fixed number of processing threads related to that (for example, reactor-http-nio- with the Reactor Netty connector). However, if Reactor Netty is used for both client and server, the two share event loop resources by default.

Reactor and RxJava provide thread pool abstractions, called schedulers, to use with the publishOn operator that is used to switch processing to a different thread pool. The schedulers have names that suggest a specific concurrency strategy — for example, “parallel” (for CPU-bound work with a limited number of threads) or “elastic” (for I/O-bound work with a large number of threads). If you see such threads, it means some code is using a specific thread pool Scheduler strategy.

Data access libraries and other third party dependencies can also create and use threads of their own.

Seessel answered 19/2, 2024 at 21:55 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.