1. 线程中断的概念与方法
中断: 每个线程有一个中断状态,且初始时,中断状态为false。
当线程A通过调用Thread.interrupt()中断一个线程B,会出现一下两种情况之一:
a、如果线程B正在执行一个低级可中断阻塞方法,如Thread.sleep(), Thread.join(), Object.wait(),则取消阻塞并且抛出InterruptedException。
b、相反,interrupt()方法只是设置线程B的中断状态为true, 当线程B询问中断状态时候(Thread.currentThread().isInterrupted()读取中断状态,并且可以通过Thread.interrupted()的操作读取和清除中断位(变为true))。
因此,当一个线程中断另外一个线程时候,被中断的线程不一定要立即停止正在做的工作
停止线程:在线程处理完任务之前,停掉正在做的工作。
Java中,3种终止正在运行的线程的方法:
使用退出标志,使线程正常退出,即run方法完成后正常停止。
使用Thread.stop() 强制终止线程,过期作废的方法,排除不用。
使用interrupt方法终止线程。
NOTE: 采用Thread.interrupt()方法,这个方法不会直接终止一个正在运行的线程,还需要加入一个判断才能够完成线程的停止。
2. interrupt 方法终止线程的具体操作
1. interrupt + for break
package com.guoqiang;
public class BreakInterrupt extends Thread {
@Override
public void run() {
super.run();
for (int i = 0; i < 50000; i++) {
if (Thread.currentThread().isInterrupted()) { // 用Thread.currentThread().isInterrupted() 方法判断当前线程是否被中断
System.out.println("要退出了,拜拜~");
break; // 使用break退出 for 循环
}
System.out.println("i = " + (i + 1));
}
System.out.println("如果此段代码在for循环中继续运行,线程就未停止~");
}
}
public class Main {
public static void main(String[] args) {
Thread breakInterrupt = new BreakInterrupt();
breakInterrupt.start();
try {
Thread.sleep(400);
breakInterrupt.interrupt();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("END");
}
}
2. 将interrupt和return结合起来 结束线程
if (Thread.currentThread().isInterrupted()) {
System.out.println("this Thread END");
return;
}
3. 中断相关函数
Thread.interrupt() :
(1)将线程的中断状态设置成为true;
(2)让被阻塞的线程抛出InterruptException异常(同时中断状态为false)
注意:没有占用CPU运行的线程是不可能给自己的中断状态置位的,就会产生一个InterruptedException异常。
比如说:线程A获得了锁进入了同步代码块中,但由于条件不足调用 wait() 方法阻塞了。这个时候,线程B执行 threadA.interrupt()请求中断线程A,此时线程A就会抛出InterruptedException,我们就可以在catch中捕获到这个异常并进行相应处理(比如进一步往上抛出)
Thread.isInterrupted():
检测调用该方法的线程是否被中断,中断状态不会被清除。线程一旦被中断,该方法返回true,而一旦sleep等方法抛出异常,它将清除中断状态,此时方法将返回false。
Thread.interrupted() : 静态方法
检测当前线程是否被中断,并且中断状态会被清除(即重置为false);由于它是静态方法,因此不能在特定的线程上使用,只能报告调用它的线程的中断状态;如果该方法被调用两次,则第二次一般是返回false,如果线程不存活,则返回false。
public class Main {
public static void main(String[] args) {
System.out.println("1: " + Thread.interrupted());
Thread.currentThread().interrupt();
System.out.println("2: " + Thread.interrupted());
System.out.println("3: " + Thread.interrupted());
}
}
OUT:
1: false
2: true
3: false