在Java中如何用interrupt优雅的结束线程

Posted warmor

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了在Java中如何用interrupt优雅的结束线程相关的知识,希望对你有一定的参考价值。

    一般来说线程执行完run()之后就自动结束了,不过有些时候我们需要线程不停的做一些事情,也就是使用while循环,那么这时候该如何停止线程呢?

    这个问题需要分情况来讨论,如果线程做的事情不是耗时的,那么只需要使用一个标志即可,具体的代码如下:

 class MyThread extends Thread   	  
        private volatile boolean isStop = false;    
        public void run() 
        	while (!isStop) 
    			System.out.println("do something");
    		
        
        public void setStop() 
    		isStop = true;
    	      

    如果需要退出时,调用setStop()即可。这里使用了一个Java关键字volatile,这个关键字的目的是如果修改了isStop的值,那么在while循环中可以立即读取到修改后的值,关于volatile的详细介绍可以参考我的另一篇博文 java线程同步volatile与synchronized

    如果线程做的事情是耗时或者说阻塞的(如调用了sleep,同步锁的wait,socket的receiver,accept等方法),那么就需要用到interrupt()了,调用该函数时会把线程设置为中断状态。如果是sleep,wait则会抛出InterruptedException异常,代码中通过捕获该异常,然后break出循环,就可以了。代码如下:

public static void main(String[] args) 
  Thread t1 = new Thread(new Runnable() 

    @Override
    public void run() 
      System.out.println("t1 run...");
      int i = 0;
      //location 2
      while (Thread.currentThread().isInterrupted() == false) 

        System.out.println("i is " + i);
        i++;
        try 
    					
          Thread.sleep(1000);
         catch (InterruptedException e) 
    	  System.out.println("this is InterruptedException");
          break; //location 1
        
      
      boolean isInterrupted = Thread.currentThread().isInterrupted();
      System.out.println("thread is interrupt ? " + isInterrupted);
    
  );
  t1.start();
  System.out.println("main run...");
  try 
    Thread.sleep(10);
   catch (InterruptedException e) 
    e.printStackTrace();
  
  t1.interrupt();

    程序很简单,运行后,启动t1线程,然后主线程sleep 10毫秒,接着执行interrupt,让t1处于中断状态。结果打印如下:

main run...
t1 run...
i is 0
this is InterruptedException
thread is interrupt ? false

    因为执行了interrupt,而t1正处于sleep状态,所以会抛出 InterruptedException 异常,捕获异常后执行break就可以跳出while循环,这样线程就可以结束了。这里需要注意的是,捕获异常的同时,程序会把线程的中断状态重置,所以下面判断的结果为false。如果这里(location 1)不执行break,那么while循环会一直执行下去。

    下面来看看 isInterrupted()和 interrupted()的区别,代码如下:

public static void main(String[] args)
  
  Thread t1 = new Thread(new Runnable() 

    @Override
    public void run() 
      System.out.println("t1 run...");
      int i = 0;
      //location 2
      while (Thread.currentThread().isInterrupted() == false) 

        System.out.println("i is " + i);
        i++;
      
      boolean isInterrupted = Thread.currentThread().isInterrupted();
      System.out.println("thread is interrupt ? " + isInterrupted);
    
  );
  t1.start();
  System.out.println("main run...");
  try 
    Thread.sleep(10);
   catch (InterruptedException e) 
    e.printStackTrace();
  
    t1.interrupt();

    这里,先启动线程t1,然后主线程sleep 10毫秒,接着执行t1的interrupt函数。结果打印如下:

main run...
t1 run...
i is 0
i is 1

。。。

i is 305
i is 306
thread is interrupt ? true

    这很容易理解,一开始t1会一直打印,直到执行了interrupt,线程就会处于中断状态,那么while的条件就不满足了,所以会跳出循环。这种情况可以用在程序不抛出 InterruptedException 异常时结束线程。

    如果把location 2处的判断条件改为 Thread.interrupted() == false ,那么会出现什么情况呢?看看打印吧!

main run...
t1 run...
i is 0

。。。

i is 372
i is 373
thread is interrupt ? false

    前面的打印都一样,只是在打印线程的状态时,出现了差别。这是什么原因呢? 原来isInterrupted()和interrupted()都可以判断线程的状态,所不同的是,interrupted()在判断完之后会把线程的状态重置,所以后面再次判断时,线程就不处于中断状态了。

    其实还有一个办法那就是使用thread.stop()来强行终止线程,不过由于该方法不安全已经废弃掉了,因为他有下面两个缺陷:

1. 立即抛出ThreadDeath异常,在线程的run()方法内,任何一点都有可能抛出ThreadDeath异常,包括在catch或finally语句中。

2. 释放该线程所持有的所有的锁。

    好了,关于线程如何停止的问题到这里就圆满收官了,大家有什么问题欢迎留言讨论哈

以上是关于在Java中如何用interrupt优雅的结束线程的主要内容,如果未能解决你的问题,请参考以下文章

Java 线程的终止-interrupt

详解线程interrupt()方法

java多线程技术: interrupt() 中断线程, 优雅停止线程及原理

java 多线程6: 中断机制 优雅的终止java线程

Java线程的中断(Interruption)

JAVA-初步认识-第十四章-多线程-停止线程方式-interrupt