I want to create pool of virtual threads running on separate pool of Java Threads.
Here's the architecture I am trying to create:
This is to enable me to create separate pools to run batch tasks in one JVM and take benefit of virtual threads with n:m mapping for each pool. So if i have 12 cores then i can create 2 ThreadPools of 6 Threads. Each pool will only execute a specific task. Each pool will have N number of VirtualThreads. So, here the mapping would be 2 pools of {N VirtualThreads -> 6 Platform Threads}.
TLDR, I want to limit the number of PlatformThreads that a pool of virtual threads can run on.
One thing i could think of is, create the thread pool and when passing in a runnable, inside the run method I could just create virtual threads but the not sure how practical is it and will i be getting the pool partition that i want. Another issue with this approach is, the virtual threads will be running in just one java thread, so no N:M mapping
Virtual threads were invented to avoid the bother you seem to be undertaking.
And, virtual threads are explicitly documented as not being intended for pooling. Like facial tissues, grab a fresh one, use it, and dispose.
Read the Java JEP, and watch videos with Ron Pressler, Alan Bateman, José Paumard, etc. to understand the purpose and nature of virtual thread technology.
You said:
… and:
Turn your thinking inside-out: Instead of creating virtual threads to run a bunch of related cascading tasks, create just one thread to do all the work in serial fashion.
If you have a series of cascading tasks, each task picking up after the previous task‘s result, then simply write wrap all that work in a single
Runnable
/Callable
. Execute that single combined task in a single new fresh virtual thread. Let that virtual thread run to completion.Let's devise a simple demo app. We have three tasks that feed into one another,
TaskA
,TaskB
, andTaskC
. They are housed together asAlphabetTask
. The result is "ABC", each letter having been added by each subtask.We establish three instances of AlphabetTask.
We submit all of those instances to an executor service. For each of those three
AlphabetTasks
, the executor assigns a fresh new virtual thread. Within each virtual thread, each of our subtasks are called in sequential order.Notice that we can use try-with-resources syntax to automatically close our executor service if its tasks complete in under a day.
Caveat: Virtual threads are appropriate to tasks where the code involves blocking, such as file I/O, network I/O, database calls, waiting on a lock, etc. Do not use virtual threads to run CPU-bound tasks such as video encoding (no blocking involved).
When run:
Beware: Output on console may not appear in chronological order when calling
System.out.println
across threads. Include and inspect timestamps if you care about order.Here is all of that code for your copy-paste convenience, to be dropped into a single
.java
file.You said:
You do not have as much control as you seem to believe. Which platform-threads run on which core at which time for what duration is all up to the host OS. And the scheduling behavior varies moment by moment depending on the current loads on the computer. At any moment, none, few, or all of your threads may be executing.