How do we correctly interrupt a thread that is executing? ?

How do we correctly interrupt a thread that is executing? ?

[[358852]]

The author has developed a simple, stable, and scalable delayed message queue framework for high-concurrency scenarios. It has precise timed tasks and delayed queue processing functions. Since it was open sourced for more than half a year, it has successfully provided precise timed scheduling solutions for more than a dozen small and medium-sized enterprises and has withstood the test of production environments. In order to benefit more friends, the open source framework address is now provided: https://github.com/sunshinelyz/mykit-delay

Preface

When we call the wait() method of a Java object or the sleep() method of a thread, we need to catch and handle the InterruptedException. If we do not handle the InterruptedException properly, unexpected consequences will occur! Today, we will use a case to explain in detail why interrupting the execution of a thread does not work.

Program Case

For example, in the following program code, the InterruptedTask class implements the Runnable interface. In the run() method, the handle of the current thread is obtained, and in the while(true) loop, the isInterrupted() method is used to detect whether the current thread is interrupted. If the current thread is interrupted, the while(true) loop is exited. At the same time, in the while(true) loop, there is a line of Thread.sleep(100) code, and the InterruptedException is caught. The entire code is shown below.

  1. package io.binghe.concurrent.lab08;
  2.  
  3. /**
  4. * @author binghe
  5. * @version 1.0.0
  6. * @description Thread test interruption
  7. */
  8. public class InterruptedTask implements Runnable{
  9.  
  10. @Override
  11. public void run() {
  12.  
  13. Thread currentThread = Thread.currentThread();
  14. while ( true ) {
  15. if(currentThread.isInterrupted()){
  16. break;
  17. }
  18.  
  19. try {
  20. Thread.sleep(100);
  21. } catch (InterruptedException e) {
  22. e.printStackTrace();
  23. }
  24. }
  25. }
  26. }

The original intention of the above code is to check whether the thread is interrupted by the isInterrupted() method, and if it is interrupted, exit the while loop. Other threads interrupt the executing thread by calling the interrupt() method of the executing thread, which sets the interrupt flag of the executing thread, so that currentThread.isInterrupted() returns true, so that the while loop can be exited.

This seems to be fine! But is it really so? We create an InterruptedTest class for testing, the code is as follows.

  1. package io.binghe.concurrent.lab08;
  2.  
  3. /**
  4. * @author binghe
  5. * @version 1.0.0
  6. * @description Test thread interruption
  7. */
  8. public class InterruptedTest {
  9. public   static void main(String[] args){
  10. InterruptedTask interruptedTask = new InterruptedTask();
  11. Thread interruptedThread = new Thread(interruptedTask);
  12. interruptedThread.start();
  13. try {
  14. Thread.sleep(1000);
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. }
  18. interruptedThread.interrupt();
  19. }
  20. }

We run the main method as shown below.

This is actually different from what we imagined! Different! Different! Why is this?

Problem Analysis

The above code clearly calls the interrupt() method of the thread to interrupt the thread, but it does not work. The reason is that when the run() method of the thread is executed, it is blocked on sleep(100) most of the time. When other threads interrupt the executing thread by calling the interrupt() method of the executing thread, it is likely to trigger InterruptedException. When the InterruptedException is triggered, the JVM will clear the interrupt flag of the thread at the same time. Therefore, the currentThread.isInterrupted() judged in the run() method will return false, and the current while loop will not be exited.

Now that the problem has been analyzed clearly, how do we interrupt the thread and exit the program?

Problem Solving

The correct way to handle it should be to reset the interrupt flag after catching the exception in the while(true) loop in the run() method in the InterruptedTask class. Therefore, the correct code of the InterruptedTask class is as follows.

  1. package io.binghe.concurrent.lab08;
  2.  
  3. /**
  4. * @author binghe
  5. * @version 1.0.0
  6. * @description Interrupt thread test
  7. */
  8. public class InterruptedTask implements Runnable{
  9.  
  10. @Override
  11. public void run() {
  12.  
  13. Thread currentThread = Thread.currentThread();
  14. while ( true ) {
  15. if(currentThread.isInterrupted()){
  16. break;
  17. }
  18.  
  19. try {
  20. Thread.sleep(100);
  21. } catch (InterruptedException e) {
  22. e.printStackTrace();
  23. currentThread.interrupt();
  24. }
  25. }
  26. }
  27. }

As you can see, we added a new line of code in the catch code block that captures the InterruptedException exception.

  1. currentThread.interrupt();

This allows us to reset the thread's interrupt flag after catching the InterruptedException, thereby interrupting the currently executing thread.

We run the main method of the InterruptedTest class again as shown below.

Summarize

Be careful when handling InterruptedException. If an InterruptedException is thrown when the execution thread is interrupted by calling the interrupt() method of the execution thread, the JVM will clear the interrupt flag of the execution thread at the same time as the InterruptedException is triggered. At this time, calling the isInterrupted() method of the execution thread will return false. At this time, the correct way to handle it is to catch the InterruptedException in the run() method of the execution thread and reset the interrupt flag (that is, in the catch code block that catches the InterruptedException, call the interrupt() method of the current thread again).

This article is reprinted from the WeChat public account "Glacier Technology". You can follow it through the QR code below. To reprint this article, please contact the Glacier Technology public account.

<<:  Software-defined home products improve remote productivity

>>:  One million new 5G base stations will be built next year. Is that a lot?

Recommend

Use Zerotier to connect remote devices to a local area network

Now that we are working from home due to the pand...

Application of load balancing technology in computing power network

Part 01, ECMP ECMP is a hop-by-hop, flow-based lo...

Now is the time to use 5G indoors

Operators have made huge investments in 5G RAN, w...

A Complete Guide to Data Center Site Selection

Project site selection is the starting point for ...

I heard that the client will disappear within three years.

[[400044]] This article is reprinted from the WeC...

How will the three major operators fight in 2022?

2022 is here. In this new year, how will the thre...

Let you know the more common Wi-Fi standards and types

Wi-Fi is an all-encompassing word. In a sense, it...