终止线程的2种方法。标记法和中断
本站寻求有缘人接手,详细了解请联系站长QQ1493399855
引言
在原来的jdk版本中:
提供了暂停,恢复,终止线程的方法,分别是suspend(),resume(),stop();
但是它们都存在缺陷,比如暂停suspend()方法在调用后,线程不会释放资源(比如:锁),而是占有着资源进入睡眠状态。stop()方法在调用后,通常不能保证线程的资源正常的释放,因为他根本没有给予线程释放资源的机会。
正因为这些方法带来的不良影响,使得它们被废除。
替代
然而聪明的程序员还是想出来了替代上述api的方法,
原始api | 替代方案 |
---|---|
暂停和恢复 | 等待-通知机制 |
终止 | 标记法和中断法 |
终止线程-法一:标记法
通过置标记为相反的布尔值,来终止线程
代码:
package Interrupt;public class InterruptThread4 {public static void main(String args[]) throws InterruptedException {ThreadTest t1=new ThreadTest();t1.start();Thread.sleep(1000);System.out.println(System.currentTimeMillis());t1.stopMe();}static class ThreadTest extends Thread{private boolean stopMe=false;public void stopMe() {this.stopMe = true;}@Overridepublic void run() {while(true){if(stopMe){System.out.println("interrupt!");break;}try {System.out.println("t1 sleep");Thread.sleep(5000);} catch (InterruptedException e) {e.printStackTrace();}Thread.yield();}System.out.println(System.currentTimeMillis());}}
}
结果:
t1 sleep
1501152592588
interrupt!
1501152596589
现象:
通过打印的时间戳对比发现,发出中断信号和线程中断相隔的时间为5s,这正是线程休眠的时间。所以通过标志来中断线程,假设线程中有sleep()和wait()方法,它达不到立即终止的作用,必须等线程恢复正常运行或者唤醒后才能终止。
终止线程-法二:中断法
Thread中断是通过内部的标记字段来进行的,线程中断并不是使线程立刻退出,而是发送给线程一个通知,至于目标线程接到通知如何处理是由线程自己决定。这是和stop()方法强行退出是不同的
api
在演示实例之前,先来了解下Thread类中断的api
api | 含义 |
---|---|
public void interrupt() | 中断线程 |
public boolean isInterrupt() | 判断线程是否中断 |
public static boolean interrupted | 判断线程是否中断,并且清除当前中断状态 |
代码
package Interrupt;public class InterruptThread3 {public static void main(String args[]) throws InterruptedException {Thread t1=new Thread(){@Overridepublic void run() {while(true){if(Thread.currentThread().isInterrupted()){System.out.println("interrupt!");break;}try {System.out.println("t1 sleep");Thread.sleep(5000);} catch (InterruptedException e) {e.printStackTrace();//设置中断状态Thread.currentThread().interrupt();}Thread.yield();}System.out.println(System.currentTimeMillis());}};t1.start();Thread.sleep(1000);System.out.println(System.currentTimeMillis());t1.interrupt();}
}
结果
1501153532961
java.lang.InterruptedException: sleep interrupted
interrupt!
at java.lang.Thread.sleep(Native Method)
1501153532963
at Interrupt.InterruptThread3$1.run(InterruptThread3.java:15)
现象
发现interrupt()几乎可以达到立即中断的效果,但是线程必须捕获InterruptedException异常这是前提条件。否则线程会永远的执行下去。
至于在catch代码块中为什么要通过Thread.currentThread().interrupt();
来重置中断状态,因为当jvm因为sleep(),wait()等需要捕获异常的方法被中断时,在抛出InterruptedException之前,它会先清除掉线程的中断标志位,使得isInterrupted()返回false;
中断的两种方法对比
如果线程中有wait()和sleep()等方法,用中断法来终止线程效果更好。
因为如果线程长久的等待下去,而没有任何其他线程对其唤醒或者长期的睡眠下去,标志法根本无法中断,而中断法可以立即中断。
最后的两点非常重要:
a.要使用interrupt()方法来中断线程,一定要在方法里捕获InterruptedException。
b.如果因为wait(),或sleep()等需要捕获InterruptedException的方法被中断时,jvm会在抛出异常之前先清除中断标记位,使得isInterrupted()返回为false。素以我们要在判断之前先重新设置好标记位
好久没写博客了,闪~