Logo

dev-resources.site

for different kinds of informations.

Mastering ExecutorService Shutdown: Tracking ThreadPool Termination

Published at
1/4/2025
Categories
java
threads
threadpool
softwaredevelopment
Author
darshitpp
Author
9 person written this
darshitpp
open
Mastering ExecutorService Shutdown: Tracking ThreadPool Termination

Let's say you want to execute some tasks. Since executing it through a single thread might take you quite some time to get the result, you decide to use the ever dependable ExecutorService to process it through multiple threads.

Here's a sample:

public static void main(String[] args) {
    ExecutorService executorService = Executors.newFixedThreadPool(5);
    for (int i = 0; i < 5; i++) {
        int temp = i;
        executorService.submit(() -> {
            task(temp);
        });
    }
    executorService.shutdown();
    System.out.println("ExecutorService is shutdown");
}

private static void task(int temp) {
    try {
        TimeUnit.SECONDS.sleep(1L);
        System.out.println("Task " + temp + " completed");
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
}
Enter fullscreen mode Exit fullscreen mode

Of course, as usual, no example of Threads is ever complete without using "sleep" as an archetype of task execution.

It outputs,

ExecutorService is shutdown
Task 1 completed
Task 2 completed
Task 0 completed
Task 4 completed
Task 3 completed
Enter fullscreen mode Exit fullscreen mode

Now imagine there's an endless queue of tasks, the number of which you don't know about. Maybe they are determined by the number of entries in a database which get added dynamically.

For example, a bank, wherein which it has to process a number of transactions throughout the day. The transaction end time will be 5 PM, beyond which it will not accept any additional tasks.

However, you do know that the number of tasks will be finite, and will end at some point of time.

How do you know the point in time when all the tasks have completed?

If you notice the above code snippet, the ExecutorService.shutdown() enables the main thread to exit immediately, but the background threads still process the accepted tasks to completion. Is there a way when you can get notified about the completion?

A couple of solutions come to mind:

  1. Use a CountDownLatch to count the tasks - but since you don't know the number of tasks, it's impractical to use it.
  2. Use ExecutorService.awaitTermination(). However, the time here is still undeterministic. You can use a very liberal ExecutorService.awaitTermination(Long.MAX_VALUE, TimeUnit.DAYS) or something similar. But that is again a blocking call.

Is there a better way to solve this?

Java does provide a better and relatively unknown way to get around this. The "trick" here is to know that Executors.newFixedThreadPool is essentially a ThreadPoolExecutor with predefined values. Let's check the implementation of Executors.newFixedThreadPool.

public static ExecutorService newFixedThreadPool(int nThreads) {
    return new ThreadPoolExecutor(nThreads, nThreads,
                                  0L, TimeUnit.MILLISECONDS,
                                  new LinkedBlockingQueue<Runnable>());
}
Enter fullscreen mode Exit fullscreen mode

I really recommend to read the doc for ThreadPoolExecutor here. ExecutorService is a convenient wrapper over ThreadPoolExecutor.

... programmers are urged to use the more convenient Executors factory methods Executors.newCachedThreadPool() (unbounded thread pool, with automatic thread reclamation), Executors.newFixedThreadPool(int) (fixed size thread pool) and Executors.newSingleThreadExecutor() (single background thread), that preconfigure settings for the most common usage scenarios.

The section that will help us resolve out problem is:

Hook methods

This class provides protected overridable beforeExecute(Thread, Runnable) and afterExecute(Runnable, Throwable) methods that are called before and after execution of each task. These can be used to manipulate the execution environment; for example, reinitializing ThreadLocals, gathering statistics, or adding log entries. Additionally, method terminated() can be overridden to perform any special processing that needs to be done once the Executor has fully terminated.

We can use the terminated method to notify us of the same! But how do we use it?

public static void main(String[] args) {
    ExecutorService executorService = new ThreadPoolExecutor(5, 5,
            0L, TimeUnit.MILLISECONDS,
            new LinkedBlockingQueue<>()) {
        @Override
        protected void terminated() {
            super.terminated();
            System.out.println("ExecutorService is terminated");
        }
    };
    for (int i = 0; i < 5; i++) {
        int temp = i;
        executorService.submit(() -> {
            task(temp);
        });
    }
    executorService.shutdown();
    System.out.println("ExecutorService is shutdown");
}

private static void task(int temp) {
    try {
        TimeUnit.SECONDS.sleep(1L);
        System.out.println("Task " + temp + " completed");
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
}
Enter fullscreen mode Exit fullscreen mode

If you do not prefer Anonymous classes (like me), you can always extend ThreadPoolExecutor yourself to create a custom one.

public static void main(String[] args) {
    ExecutorService executorService = getThreadPoolExecutor();
    for (int i = 0; i < 5; i++) {
        int temp = i;
        executorService.submit(() -> {
            task(temp);
        });
    }
    executorService.shutdown();
    System.out.println("ExecutorService is shutdown");
}

private static ThreadPoolExecutor getThreadPoolExecutor() {
    return new CustomThreadPoolExecutor(5, 5,
            0L, TimeUnit.MILLISECONDS,
            new LinkedBlockingQueue<>());
}

static class CustomThreadPoolExecutor extends ThreadPoolExecutor {
    public CustomThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue) {
        super(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue);
    }

    @Override
    protected void terminated() {
        super.terminated();
        System.out.println("ExecutorService is terminated");
    }
}

private static void task(int temp) {
    try {
        TimeUnit.SECONDS.sleep(1L);
        System.out.println("Task " + temp + " completed");
    } catch (InterruptedException e) {
        throw new RuntimeException(e);
    }
}
Enter fullscreen mode Exit fullscreen mode

Here's the output to verify if it works as per our expectations.

ExecutorService is shutdown
Task 0 completed
Task 3 completed
Task 2 completed
Task 4 completed
Task 1 completed
ExecutorService is terminated
Enter fullscreen mode Exit fullscreen mode

What are some other relatively unknown snippets you use? Let me know in the comments!

threads Article's
30 articles in total
Favicon
Launch Announcement: TwistFiber - Automate Your Threads Workflow 🚀
Favicon
Mastering ExecutorService Shutdown: Tracking ThreadPool Termination
Favicon
Threads Overhauls Its Search Feature to Improve User Experience and Engagement
Favicon
Concurrency in Python with Threading and Multiprocessing
Favicon
The Benefits of Having More Threads than Cores: Unlocking the Power of Multi-threading in Modern Computing
Favicon
What is Python GIL? How it works?
Favicon
🚀 Edge Detection with Threads and MiniMagick in Ruby 🌄
Favicon
Master OS Concepts 💡 | Understanding Threads, Processes & IPC 🔧
Favicon
Join Our Threads Community for Exclusive Bad Bunny Merch Discussions!
Favicon
Threads-API ist da
Favicon
Connect with NBA YoungBoy Merch on Threads!
Favicon
Estudo de caso: Thread pools e Out-of-memory
Favicon
Paralelismo e Concorrência 102: Java parallel streams na prática
Favicon
Concurrency and Parallelism in Ruby
Favicon
My take on Modeling Large Amounts of HLA Molecules with Ruby
Favicon
In Java what is ConcurrentModificationException? How to avoid it in multi-threading. #InterviewQuestion
Favicon
Threads API is here
Favicon
How Threads and Concurrency Work in Linux Systems
Favicon
Mastering Concurrency in Go: A Comprehensive Guide
Favicon
Is Threads still a thing?
Favicon
Async/Await: O que tem de novo no .NET 8?
Favicon
How Buying Threads Likes Can Skyrocket Your Follower Count
Favicon
Async/Await: Para que serve o CancellationToken?
Favicon
#6 Subgrid, Threads, Clouds Aren't Real, and Intl.Segmenter
Favicon
Threads Video Downloader - Snapthreads.io
Favicon
Demystifying Java Threads for Beginners
Favicon
Explorando o Multitarefa com Threads em Java
Favicon
Você já usou o System.Threading.Channels?
Favicon
Multithreading in Python: the obvious and the incredible
Favicon
Parallelism via Fiber Coroutines

Featured ones: