Thread: Smallest unit of execution that can be scheduled to the CPU.
Max number of threads that can run parallely at a given point in time = Number of CPU cores, other threads are blocked.
Process: Group of associated threads. It comprises of multiple threads and a shared memory. Single unit of functionality. A program in execution.
Task: Single unit of work performed by a thread.
Concurrency: Multiple tasks being executed at the same time. The CPU time is divided between each via Scheduling. (Not to be confused with Parallel processing)
Context switch: Storing state of a thread and later restoring it. Lesser the total context switches, the better.
Analogy - Google Chrome browser is a process, each Tab is a thread and context switching occurs when we switch tabs.
Thread priority: It is a number associated with a thread that the scheduler uses to schedule. In Java, it can be either of these three integer values - MIN_PRIORITY
(1), NORM_PRIORITY
(5), MAX_PRIORITY
(10).
Everything in Java runs on a single thread (hence synchronously) called main
unless we use concurrency features like Thread, Concurrency API, CompletableFuture
, etc.
Tasks are defined using the Runnable
functional interface, it takes no arguments and returns nothing. To create a thread use Thread
class instance.
Runnable t = () -> System.out.println("Hello"); // any returned value from a Runnable is ignored
new Thread(t).start();
System.out.println("World");
// the order of execution upon creation of a thread on line 1 depends upon the scheduler and can't be predicted beforehand
// can print "HelloWorld" or "WorldHello" because there are two threads - the one we created and one having main()
// start() creates a separate call stack for the thread execution
// calling start() implicitly calls run()
// Also if we use run() on line 2 above, it won't start a new thread and moreover it will just execute task in a non-concurrent synchronous way in main() thread only printing "HelloWorld"
// calling start() twice on same thread throws IllegalStateException, but we can call run() multiple times
Threads always proceed asynchronously with each other. Demonstrated in below example.
new Thread(()->{
for(int i = 0; i < 5; i++){
System.out.println(i);
}
}).start();
new Thread(()->{
System.out.println("A");
System.out.println("B");
System.out.println("C");
System.out.println("D");
}).start();
// 3 threads - main() and 2 new threads we created
// output unkown until runtime, one possible outcome is:
/*
1
A
2
3
B
C
4
D
*/
// use t.join() to stop the current thread and wait for thread "t" to finish
Runnable task = () -> System.out.println("Hello");
Thread thread = new Thread(task);
thread.start();
thread.join();
System.out.println("World");
// output:
/*
Hello
World
*/
A Class is used to create a Thread. Two ways to create:
Runnable
interface in your class: override (implement) run()
method and call the the Thread class constructor by supplying this class object as a parameter (recommended)Thread
class: override run()
method and call start()
on the class object (limited because no multiple inheritance allowed in Java)// 1
class Foo implements Runnable{
@Override
public void run(){
System.out.println("foo");
}
}
// 2
class Bar extends Thread{
@Override
public void run(){
System.out.println("bar");
}
}
// in main()
Runnable task = new Foo();
Thread t1 = new Thread(task);
t1.start();
Thread t2 = new Bar();
t2.start();
Things to do in the thread go inside the run()
method. It is the only method present in the Runnable
interface. The Thread
class itself implements Runnable
interface and that’s why it has the run()
method available to it by default.
System threads: Created by JVM. Like garbage collection.
User-defined threads: Created by the programmer.
They both can be created as daemon threads which doesn’t prevent JVM from exiting when the program finishes. Ex - Garbage collection is a daemon thread and JVM can exit even if its running.
var job = new Thread(() -> foobar());
job.setDaemon(true); // create as daemon thread
job.start(); // we won't wait until foobar() finishes, we will exit once main() finishes
var job = new Thread(() -> taskMethod());
job.getState(); // prints thread state (RUNNABLE, WAITING, etc...)
Thread.sleep(1_000); // sleeps for 1 sec, resumes thread execution upon timeout
// any other thread can interrupt a sleeping thread, in that case InterruptedException is thrown
// also, an already interrupted thread (isInterrupted() = true) can't sleep (throws InterruptedException)
var t2 = Thread.currentThread(); // get instance of current thread
t2.interrupt(); // used to throw InterruptedException in a sleeping thread
// calling interrupt() on a RUNNABLE thread (either self or from another thread) doesn't change its state (no halts or exceptions) but changes value on:
t2.isInterrupted(); // returns true
Interrupting a running (RUNNABLE
) thread with interrupt()
:
class Test extends Thread {
@Override
public void run() {
System.out.println("A");
var t = Thread.currentThread();
System.out.println(t.isInterrupted());
t.interrupt(); // self-interruption
System.out.println("B");
System.out.println(t.isInterrupted());
System.out.println("C");
}
}
public class Main {
public static void main(String[] args) {
var t = new Test();
t.start();
System.out.println("Z");
System.out.println("Y");
System.out.println("X");
}
}
/*
Z
Y
X
A
false
B
true
C
*/
Interrupting a sleeping thread (TIMED_WAITING
) with interrupt()
:
class Test extends Thread {
@Override
public void run() {
System.out.println("A");
try {
Thread.sleep(5000); // 5 sec
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
System.out.println("B");
System.out.println("C");
}
}
public class Main {
public static void main(String[] args) {
var t = new Test();
t.start();
t.interrupt(); // called within 5 sec after thread start
System.out.println("Z");
System.out.println("Y");
System.out.println("X");
}
}
// InterruptedException is thrown
java.util.concurrent
package is the concurrency API. It provides higher level abstractions, rather than working with threads directly.
We use ExecutorService
inteface to create and manange threads.
It also provides a Executors
factory class.
ExecutorService service = Executors.newSingleThreadExecutor();
try{
service.execute(t1); // t1, t2, and t3 are Runnable types (tasks)
service.execute(t2);
service.execute(t3);
}
finally{
service.shutdown();
}
// this only creates two threads - one main() and one new thread. The intra-thread tasks are guaranteed to be executed sequentially in the order of submission and won't interfere with each other as was shown in the example at the beginning with Thread class.
The shutdown()
method is of significance because SingleThreadExecutor
is a non-daemon thread so our program will never terminate if it is not closed. After a call to shutdown()
the service won’t accept any new tasks and if added will throw RejectedExecutionException
. The service is terminated after all existing tasks have finished.
The compiler won’t throw any exception when we don’t add a shutdown()
, but in that case our program will never terminate. So, always remember to call shutdown()
. It is the reason we put it in a finally
block, so that it always executes.
Do note that the service won’t stop executing already added tasks upon a shutdown()
call. To do so we can call shutdownNow()
method but nothing is guranteed.
Also, ExecutorService
interface doesn’t extend AutoCloseable
so we can’t use try-with-resources with it.
We can also await termination after a shutdown call and check on termination status.
try{
service.execute(t1);
}
finally{
service.shutdown();
}
service.awaitTermination(1, TimeUnit.MINUTES);
if(service.isTerminated()){
System.out.println("Done!");
}
else{
System.out.println("Not completed yet!");
}
The submit()
is preferred over execute()
because it is compatile with both Runnable
and Callable
and it returns a Future<?>
reference with which we can track the result.
Future<?> result = service.submit(t1); // Future ref's formal type is whatever type t1 returns
// instance methods of Future<>
result.isDone() // returns true if completed, exception or cancelled
result.isCancelled() // returns true if cancelled
result.cancel() // attempts to cancel and returns true if successfully cancelled
result.get() // get result or wait endlessly (indefinite blocking)
result.get(10, TimeUnit.SECONDS) // get result or wait for specified time, throw TimeoutException upon timeout
It is preferred over Runnable
because its call()
method can return a value and can throw a checked exception. We often use Callable
type with submit()
and this leads to clean code.
The submit()
method can also take Runnable
but calling get()
on it will always return null
since return type of Runnable’s run()
returns void
.
Future<Integer> result = service.submit(() -> 30 + 11);
System.out.println(result.get()); // 41
// todo the above with Runnable and execute() will be too complex as we will need a common variable and polling with sleep or an interrupt.
ScheduledExecutorService
is a sub-interface of ExecutorService
. The schedule methods in this return ScheduledFuture
reference which is similar to Future
and additionally has a getDelay()
method to return remaining delay.
ScheduledExecutorService service = Executors.newSingleThreadScheduledExecutor();
Runnable task1 = () -> System.out.println("Hello Zoo");
Callable<String> task2 = () -> "Monkey";
ScheduledFuture<?> r1 = service.schedule(task1, 10, TimeUnit.SECONDS);
ScheduledFuture<?> r2 = service.schedule(task2, 8, TimeUnit.MINUTES);
// The first task is scheduled 10 seconds in the future, whereas the second task is scheduled 8 minutes in the future
// if service is shutdown before they actually get to execute, then they will be discarded
service.scheduleAtFixedRate(task, 5, 1, TimeUnit.MINUTES);
// submits task after initial delay of 5 mins, and keeps submitting after 1 min intervals regardless of previous task finished
// be careful as the tasks can pile up if period is less than execution time of the task
service.scheduleWithFixedDelay(task1, 0, 2, TimeUnit.MINUTES);
// the 2 min delay is after the finish of the previous task unlike above method
Pools have pre-instantiated threads ready to run tasks. The difference between single thread executor and pools is that pools can execute multiple threads concurrently and if entire pool threads are occupied, the executor waits for threads to become available.
ExecutorService
newSingleThreadExecutor()
ScheduledExecutorService
newSingleThreadScheduledExecutor()
ExecutorService
newCachedThreadPool()
/* Creates thread pool that creates new threads as
needed but reuses previously constructed threads
when they are available. */
ExecutorService
newFixedThreadPool(int)
/* Creates thread pool that reuses fixed number of
threads operating off shared unbounded queue. */
ScheduledExecutorService
newScheduledThreadPool(int)
/* Creates thread pool that can schedule commands
to run after given delay or execute periodically. */
A lot of times threads may share a common variable that they read from or write to during parallel executions.
Consider a ++variable
operation runnning in multiple threads concurrently and on the same variable.
int x = 0;
public void incrementX(){
Syste.out.println(++x);
}
ExecutorService service = Executors.newFixedThreadPool(20);
for(int i = 0; i < 10; i++){
service.submit(() -> incrementX());
}
// can print in any order; even duplicates, because 10 threads will access x and overwrites will happen
Declaring the variable as volatile
indicates not to store the value in that thread’s own cache but on a shared cache accessible by all threads. As a result, any write to this variable will immediately be visible to all the other threads.
But it won’t make the code thread-safe because our individual read/write are a thread-safe, but ++x
is actually two operations x = x + 1
(add and assign) and it may happen that thread1 writes to x
and thread2 reads from x
and then thread1 writes to x
.
volatile int x = 0;
Atomic classes provided by Java makes sure that an atomic instance is accessed only by a single thread at a given time and all operations are atomic i.e. executed as a single unbreakable unit.
AtomicInteger x = new AtomicInteger(0);
x.incrementAndGet(); // equivalent to ++x
// other methods
x.set(9); // assignment
x.getAndIncrement(); // x++
Now, our code will output numbers from 0 to 10 with no duplicates but they may or may not be in order. Because operations are atomic now but we haven’t specified any order of execution (incrementing variable x) among threads. A thread that outputs 9
can print after a thread that prints 10
.
Monitor: also called a Lock or Mutex is a structure that supports mutual exclusion, a property that atmost one thread can be executing a code block at a given time.
Objects can be used as a lock in Java in synchronized
blocks. All threads must be accessing the same Object and which object it is doesn’t even matter as long its the same Object, preferrably final
so it doesn’t change values as we use it for sync.
// on an Object - block
synchronized(this) { }
// on method
synchronized void foobar() { }
-----------------------------------
// in static method
synchronized(Foobar.class) { } // since we can't use "this" object here; so we use class object
Test t = new Test();
synchronized(t) { } // create and use any random instance
// on static method
static synchronized void foobar() { }
When a thread arives at the block, it checks the lock status. If the lock isn’t available, the thread will transition to a BLOCKED
state and waits in queue otherwise it “acquires the lock” and enters the code block.
This will print numbers from 0 to 10 in order without duplicates.
Syncs on a Lock
interface object instead of any Object. We have to release lock manually with unlock()
or else other threads keep on waiting forever. A thread can acquire a lock multiple times (re-entry) but it itself must release it the same number of times (hold count) as other threads can’t release a lock acquired by a thread.
Lock lock = new ReentrantLock();
try{
lock.lock(); // waits/blocks indefinitely if other thread has lock
// code
}
finally{
lock.unlock();
}
// above code in synchronized block
Object obj = new Object();
synchronized(obj){
// code
}
// constructor of ReentrantLock
Lock lock = new ReentrantLock(true); // fairness property
// true - grant lock in order it was requested
// false - default no-arg call
// attempting to unlock a lock which was never acquired leads to
IllegalMonitorStateException
If we attempt to lock()
, but it is already acquired by any other thread, then lock()
will keep on waiting indefinitely for the lock to become available to it.
tryLock()
returns a boolean
indicating the status of lock and tryLock(long, TimeUnit)
will attempt to lock at specified intervals each time returning a boolean
. Also, both of them are non-blocking unlike lock()
and return a boolean
immediately instead of waiting indefinitely.
// code demo
public static void printHello(Lock lock) {
try {
lock.lock();
System.out.println("Hello");
}
finally {
lock.unlock();
}
}
public static void main(String[] args) {
Lock lock = new ReentrantLock();
new Thread(() -> printHello(lock)).start(); // sending lock to be acquired by a thread
if(lock.tryLock()) { // tryLock() attempting to lock to Main thread
try {
System.out.println("Lock obtained, entering protected code");
}
finally {
lock.unlock();
}
} else {
System.out.println("Unable to acquire lock, doing something else");
}
}
NOTE: Release a lock the same number of times it is acquired by the same thread. Otherwise, the remaining locks will still hold onto the thread.
Semaphore: signaling mechanisms for restricting access to a single resource, maintains a set of permits, they permit fixed number of multiple threads to enter critical section. They are used to limit the number of concurrent threads accessing a specific resource (like a funnel).
Semaphore smp = new Semaphore(3);
// permits 3 threads to acquire it simultaneously; others get blocked in a queue
public static void printHello(Semaphore smp) {
try {
smp.acquire();
// code
}
finally {
smp.release();
}
}
new Semaphore(1)
is a semaphore that has only two states: 1
permit available or 0
permits available. Permit count is exactly 1.Are Locks and Binary Semaphores interchangeably used? Yes, they can be. But there are significant diff between the two:
Orchestrating tasks, specifies the number of threads to wait for and once the number is reached, execution is resumed on all of the threads that the barrier was “holding”.
var c1 = new CyclicBarrier(5);
// in code executed by multiple threads
c1.await()
// keeps waiting until 5 threads are in waiting status, then resumes on all
// throws
InterruptedException
BrokenBarrierException
// if we have 15 threads and barrier size is 5 then -
// await 5 threads and break barrier
// await next 5 and break barrier
// await next 5 and break barrier
// so its called three times, and everytime threads resume in set of 5
CountDownLatch
is another synchronization aid and its instance has a counter field, which we decrement manually using countDown()
method. The current thread gets blocked until counter reaches 0
. Note that we manually decrease the counter after each “task” completion and it’s not necessarily the thread count like a CyclicBarrier
.
// in main method
CountDownLatch countDownLatch = new CountDownLatch(5); // create latch
foobar(1, countDownLatch);
// call foobar() and wait for all tasks to complete
countDownLatch.await();
// task method
void foobar(int x, CountDownLatch countDownLatch){
// do some processing and after that dec counter
countDownLatch.countDown();
}
All threads must have a consistent view of a collection at all times. Therefore any kind of collection (non-sync/sync/concurrent) can be read from across threads unless there is modification to it in real-time.
Even if used in a single thread, we can have ConcurrentModificationException
at runtime when we try to modify a non-concurrent collection from within that thread itself while iterating on it (Fail-Fast iterators). So when dealing with threads, always use concurrent collections.
// non-concurrent collection
Map<String, String> mp = new HashMap<>();
// use a concurrent collection - multiple threads at a time can modify
Map<String, String> cmp = new ConcurrentHashMap<>();
// or make an existing one synchronized - single thread at a time can modify
Collections.synchronizedMap(mp);
// or make an existing one immutable - none can modify
Collections.unmodifiableMap(mp);
If we have to create a collection, we use concurrent version of collections available. If we have a non-conurrent collection then we can use synchronized methods from Collections
class to make them compatible to threads using synchronized methods that exists for most collections.
There is a diffrence between synchronized and concurrent collections though! An existing collection made synchronized can only be accessed by only a single thread at a time (acquires lock on the entire object). A concurrent collection is superior, it will allow collection’s access by multiple-threads at the same time by dividing and assigning segments to each thread trying to access it (acquires locks on separate segments of object).
Immutable collections are immune to memory inconsistency errors trivially.
Any concurrent collection with Skip
in the name is sorted. CopyOnWrite
concurrent collections are deeply copied everytime on add/remove/change and assign to original reference.
Liveness is the ability of an application to execute in a timely manner. If it becomes unresponsive for long and or keeps entering “stuck” state repeatedly, liveness is affected. Threads can enter WAITING
or BLOCKED
state and cause liveness issue.
Majorly 3 kinds of liveness affecting issues arise: Deadlock, Starvation, and Livelock.
Race condition: when a shared resouce is modified by two threads at the exact same time. The application has to decide which modification to keep or discard both. It often leads to invalid data when it occurs.
Two or more threads being blocked forever, each one waiting for other to release resources.
// first thread enters this; after acquiring lock for obj1
synchronized (obj1){
synchronized (obj2){ }
}
// second thread enters this; after acquiring lock for obj2
synchronized (obj2){
synchronized (obj1){ }
}
// both keep waiting for each other to release locks on thier respective nested sync block's obj
// so that they can acquire lock and enter nested sync block but it never happens - circular waiting
Identify: Use jstack
or jvisualvm
to generate Thread Dump and analyze it to identify deadlocks.
Resolve: avoid nested locks, order locks properly, use ReentrantLock
avoiding usage of synchronized
blocks as they offer tryLock()
with timeout.
A thread perpetually keeps waiting for a resource to become available. Unresolved deadlocks always lead to the starvation of both the threads involved.
A special kind of deadlock in which threads appear to be live (changing states) but actually they are just stuck in a loop, unable to make progress, effectively stuck in an unproductive cycle of reacting to each other.
Ex - two people attempting to pass each other in a corridor.
Serial streams: Data is processed in a serial fashion one after the other.
Parallel streams: Multiple parallel threads can process data concurrently. The number of threads available are common for all streams in the program and come from a Fork-Join Thread Pool.
As with threads, the order of operations is never guaranteed in parallel streams.
// making a Stream parallel
collection.stream().parallel();
collection.parallelStream();
// checking if a stream is parallel
stream.isParallel();
When we apply methods which are order dependent (findFirst()
, limit()
, skip()
) on parallel stream, we do get what we expect just like in serial stream but performance is hampered since stream is forced by Java to be synchronized-like.
Solution - use unordered()
to declare stream as unordered, and avoid force conversion of parallel to serial when such methods are applied.
stream.unordered();
Calling unordered()
(intermediate operation) on serial stream has no effect, but when the stream is made parallel we will have skip(5)
skipping any 5 random elements and not the first five.
All parallel streams in a program use a common Fork-Join Thread Pool, and if you run a long-running task in the stream, you’re effectively holding multiple threads in the pool. Consequently, you may end up blocking all other tasks in the program that are unrelated but are using parallel streams if all ForkJoinPool threads become busy!
List<StockInfo> getStockInfo(Stream<String> companies) {
return companies.parallel()
.map(this::getStockInfo) // slow operation; calls method on every element
.collect(Collectors.toList());
}
// this hangs (blocks) every other method in execution that is using parallel streams if enough threads aren't available in the pool
// not only "companies" stream, but any parallel stream part of the code flow!
By default, Number of threads in common ForkJoinPool = Number of logical CPU cores - 1
, it is a fixed sized pool and other tasks have to wait till threads become available. We can customize the size of pool of course.
Make sure accumulator and combiner have the same output on every step regardless of the order in which they are called in. The accumulator and combiner must be associative, non-interfering, and stateless.
Virtual threads are lightweight threads that can be used in high-throughput concurrent applications.
Uptil now a thread was an instance of java.lang.Thread
and was called a Platform Thread in Java. It maps to the underlying OS thread for its whole lifetime and contains information in the order of megabytes. Consequently, the number of available platform threads is limited to the number of OS threads in a thread-per-request style server environment.
A Virtual Thread is also an instance of java.lang.Thread
, but it isn’t tied to a specific OS thread. A virtual thread still runs code on an OS thread. However, when code running in a virtual thread calls a blocking I/O operation, the Java runtime suspends the virtual thread until it can be resumed. The OS thread associated with the suspended virtual thread is now free to perform operations for other virtual threads.
When a virtual thread is blocked, its data is stored (unmounted) on the heap temporarily, until it goes back into execution on the carrier thread.
They have much smaller memory footprint than a platform thread, thus a single JVM might support millions of virtual threads.
Runnable task = () -> System.out.println("Hello");
// using Thread class
Thread virtualThread = Thread.startVirtualThread(task);
// another way
Thread thread = Thread.ofVirtual().unstarted(task);
thread.start();
// using concurreny API
ExecutorService service = Executors.newVirtualThreadPerTaskExecutor();
// creates new virtual thread for each submitted task
thread.isVirtual(); // tells if the thread is virtual
Virtual Threads have these fixed properties:
NORM_PRIORITY
(5)Virtual threads are useful when the number of concurrent tasks is large, and the tasks mostly block on network I/O. They offer no benefit for CPU-intensive tasks. Since CPU cores will be busy doing the intensive task and will block the core for everyone (other virtual threads can’t use it meanwhile) unlike Network I/O blocks.
Benefit of using Virtual Threads: Instead of writing non-blocking, asynchronous style code with CompletableFuture
we can write “normal” synchronous style code that uses blocking I/O. Virtual Threads will make sure that blocking I/O on a thread doesn’t hog resources. When a thread is ready to be resumed (wait over) it is resumed instantly without the need of specifying any callbacks.
References:
Future<T>
was introduced earlier in Java and needed more functionality (like non-blocking operations, chaining, callbacks, and combining multiple futures) so CompletableFuture<T>
was added to Java.
CompletableFuture is a framework to execute code asynchronously and return a reference immediately. It is named so because we can explicitly complete it by using complete(T t)
method on its ref and it returns t
as its completion result.
By default, it uses the common ForkJoinPool
threads just like Streams (they are Daemon Threads). We can provide a custom ExecutorService
to it too (they are User Threads).
class CompletableFuture<T> implements Future<T>
so we can store output of function in CompletableFuture too as shown in the examples below:
// perform a computation async - takes in a Supplier
CompletableFuture<String> completableFuture = CompletableFuture.supplyAsync(() -> "Hello");
// get value from future instance; it is a blocking call as it waits for return value
completableFuture.get();
// run logic without any return - - takes in a Runnable
CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
System.out.println("Foo");
System.out.println("Bar");
});
// explicitly complete the future and get value (can be used in error handling scenarios)
completableFuture.complete("Done!");
// to process the result of a computation - feed it to a function or consumer (chaining)
// note that these are non-blocking just like supplyAsync() and do not block the calling thread
// feed result of async computation into a Function
CompletableFuture<String> future = completableFuture.thenApply(s -> s + " World");
// feed to a Consumer and it returns a Void placeholder type
CompletableFuture<Void> future = completableFuture.thenAccept(s -> System.out.println("Computation returned: " + s));
// nothing to feed, nothing to get back; print a line in the console on future.get() call
CompletableFuture<Void> future = completableFuture.thenRun(() -> System.out.println("Computation finished."));
// provide a callback to run upon completion using whenComplete method (non-blocking)
completableFuture.whenComplete((result, exception) -> {
if (exception == null) {
// the computation completed normally
System.out.println("Result: " + result);
} else {
// the computation completed with an exception
System.err.println("Exception: " + exception.getMessage());
}
});
// avoid doing a blocking operation like get() where you don't want to wait. Use chaining and callbacks to process async computation's result.
Stop execution of current thread and wait for all futures to complete:
// blocking operation
CompletableFuture<Void> combinedFuture = CompletableFuture.allOf(cf1, cf2, cf3);
Specify our own ExecutorService so it doesn’t use common ForkJoinPool
:
ExecutorService exec = Executors.newFixedThreadPool(20);
CompletableFuture<String> comFut = CompletableFuture.supplyAsync(task, exec);
Why is blocking tricky with async processing? Because we may end up blocking where we shouldn’t and not blocking where we should’ve.
One scenario is when we run CompletableFuture async thread and immediately block right after that to get result value (e.g. calling get()
), then we aren’t really going async. We should use chaining and callbacks to process the result.
Another scenario can be if all user threads exit before async processing in a daemon thread completes. Consider the following example:
CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
// System.out.println(Thread.currentThread().isDaemon());
Thread.sleep(2000);
System.out.println("A");
Thread.sleep(2000);
System.out.println("B");
Thread.sleep(2000);
System.out.println("C");
});
System.err.println("Exiting...");
// Output: "Exiting..." and then process finishes with exit code 0!
// Reason: the async thread is from ForkJoinPool and its a daemon thread. If main thread exits first, JVM forces daemon threads to terminate too.
To solve this issue, we can provide a custom ExecutorService
thread pool to make async thread as a User thread.
Another solution is to either wait sometime and let the async daemon thread complete (sleep()
) or block the main thread indefinitely (get()
or join()
) until async daemon thread completes:
CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
// System.out.println(Thread.currentThread().isDaemon());
Thread.sleep(2000);
System.out.println("A");
Thread.sleep(2000);
System.out.println("B");
Thread.sleep(2000);
System.out.println("C");
});
System.err.println("Exiting...");
// Thread.sleep(10000);
future.join();
/* Output:
Exiting...
A
B
C
*/
Reference: