Java 多线程 interrupt方法
interrupt
下面是interrupt方法的文档的一部分:
* <p> If this thread is blocked in an invocation of the {@link
* Object#wait() wait()}, {@link Object#wait(long) wait(long)}, or {@link
* Object#wait(long, int) wait(long, int)} methods of the {@link Object}
* class, or of the {@link #join()}, {@link #join(long)}, {@link
* #join(long, int)}, {@link #sleep(long)}, or {@link #sleep(long, int)},
* methods of this class, then its interrupt status will be cleared and it
* will receive an {@link InterruptedException}.
View Doc
如果线程由于调用Object类的wait()方法、Thread类的join()实例方法以及Thread.sleep()静态方法而被挂起,此时其他线程调用了这个线程的interrupt方法,那么这个线程的中断状态会被清除,并会抛出一个InterruptedException异常。
从上面加粗的文字中可以得出两点结论:
1 即便调用了interrupt方法,之后用isInterrupted()方法检查它的中断状态时也不一定能得到true。
2 如果线程当前运行处的代码块不对InterruptedException异常进行合适的处理,那么interrupt方法就没有任何效果。
举一个例子来验证上面的结论:
public abstract class StoppableThread extends Thread {
@Override
public void run() {
while(!isInterrupted()){
System.out.println("isInterrupted: " + isInterrupted());
doWork();
}
System.out.println("END");
}
protected abstract void doWork();
public static void main(String[] args) {
StoppableThread thread = new StoppableThread() {
@Override
protected void doWork() {
System.out.println("running...");
try {
sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
thread.start();
try {
Thread.sleep(2200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("即将调用interrupt()方法");
thread.interrupt();
}
}
输出结果:
isInterrupted: false
running...
isInterrupted: false
running...
isInterrupted: false
running...
isInterrupted: false
running...
isInterrupted: false
running...
即将调用interrupt()方法
isInterrupted: false
running...
java.lang.InterruptedException: sleep interrupted
at java.lang.Thread.sleep(Native Method)
at readerandwriter.StoppableThread$1.doWork(StoppableThread.java:22)
at readerandwriter.StoppableThread.run(StoppableThread.java:9)
isInterrupted: false
running...
isInterrupted: false
running...
InterruptedException
interrupt方法所做的仅仅是设置了Thread对象的中断状态,InterruptedException是在线程的wait()、sleep()、join()等方法中抛出的。
修改上面的那个类的doWork()方法,让它声明抛出InterruptedException,并把实现中的sleep(500)去掉。
public abstract class StoppableThread extends Thread {
@Override
public void run() {
while(!isInterrupted()){
System.out.println("isInterrupted: " + isInterrupted());
try {
doWork();
} catch (InterruptedException e) {
System.out.println("InterruptedException is Thrown");
}
}
System.out.println("END");
}
protected abstract void doWork() throws InterruptedException;
public static void main(String[] args) {
StoppableThread thread = new StoppableThread() {
@Override
protected void doWork() throws InterruptedException{
System.out.println("running...");
}
};
thread.start();
try {
Thread.sleep(2200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("即将调用interrupt()方法");
thread.interrupt();
}
}
输出结果:
isInterrupted: false
running...
isInterrupted: false
running...
isInterrupted: false
running...
isInterrupted: false
running...
isInterrupted: false
running...
即将调用interrupt()方法
isInterrupted: false
running...
END
虽然线程结束了,但是InterruptedException异常并没有被抛出(如果抛出了会输出一行InterruptedException is Thrown)。线程是因为isInterrupted()返回true才退出循环的。结合上面interrupt方法的文档,可以得出两条结论:
1 interrupt仅仅设置了Thread对象的中断状态,InterruptedException是在线程的wait()、sleep()、join()等方法中抛出的,并且抛出后会清除掉Thread对象的中断状态。
2 当线程正在执行一些常规方法时,即便调用了interrupt方法也不会抛出InterruptedException,但会导致isInterrupted()方法返回true。
Thread.interrupted
Thread类的静态方法interrupted()用来检查当前线程的中断状态。这个方法除了返回当前线程的中断状态(true/false),还会把中断状态给清除掉。也就是说,如果连续调用两次这个方法,那么第二次必然返回的是false,因为无论中断状态原本是什么,它都已经被第一次的Thread.interrupted()清除了。
如何安全可靠地中断线程
没有一个完美的方法能够实现在调用某个方法的瞬间就让线程退出。
一个比较好的方法是在Thread子类中添加一个标志位,这里命名为isShutdownRequested,默认为false,并提供一个用于在interrupt的同时设置它为true的方法。最后,在线程的run()方法的while循环控制条件中利用这个标志位判断线程是否被要求中止。
下面提供一个StoppableThread实现,里面包含一个shutdown()方法,用于中止线程:
public abstract class StoppableThread extends Thread {
private volatile boolean isShutdownRequested = false;
public final void shutdown(){
isShutdownRequested = true;
interrupt();
}
public final boolean isShutdownRequested() {
return isShutdownRequested;
}
@Override
public void run() {
try {
while(!isShutdownRequested()){
doWork();
}
} catch (InterruptedException e) {
System.out.println("InterruptedException is Thrown");
} finally {
doShutdown();
}
}
protected abstract void doWork() throws InterruptedException;
protected abstract void doShutdown();
public static void main(String[] args) {
StoppableThread thread = new StoppableThread() {
@Override
protected void doWork() throws InterruptedException {
System.out.println("running...");
sleep(500);
}
@Override
protected void doShutdown() {
System.out.println("shutdown!");
}
};
thread.start();
try {
Thread.sleep(2200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("即将调用shutdown()方法");
thread.shutdown();
}
}
输出结果:
running...
running...
running...
running...
running...
即将调用shutdown()方法
InterruptedException is Thrown
shutdown!
参考资料
Java 多线程 interrupt方法的更多相关文章
- java多线程 interrupt(), interrupted(), isInterrupted()方法区别
interrupt()方法: 作用是中断线程. 本线程中断自身是被允许的,且"中断标记"设置为true 其它线程调用本线程的interrupt()方法时,会通过checkAcces ...
- Java——多线程之方法详解
Java多线程系列文章是Java多线程的详解介绍,对多线程还不熟悉的同学可以先去看一下我的这篇博客Java基础系列3:多线程超详细总结,这篇博客从宏观层面介绍了多线程的整体概况,接下来的几篇文章是对多 ...
- Java 多线程 sleep方法与wait方法的区别
sleep方法会使线程暂停执行一段时间,wait方法会阻塞线程,直到被唤醒或等待时间超时. 两者区别具体如下: 1 原理不同 sleep方法是Thread类的静态方法,使线程暂停执行一段时间,等到计时 ...
- 多线程——interrupt方法
测试interrupt()方法: package day_12_01_Thread; import java.util.Date; /** * 测试interrupt()方法:结束线程,但是线程还是活 ...
- Java多线程-run方法与start方法的区别
package com.interview; /** * java多线程的两种实现方式以及run.start方法的区别 * @author MEI.LIU * */ public class Thre ...
- Java多线程-join方法
thread.Join把指定的线程加入到当前线程,可以将两个交替执行的线程合并为顺序执行的线程.比如在线程B中调用了线程A的Join()方法,直到线程A执行完毕后,才会继续执行线程B. 具体例子看链接 ...
- 多线程 interrupt()方法
java interrupt()方法只是设置线程的中断标记,当对处于阻塞状态的线程调用interrupt方法时(处于阻塞状态的线程是调用sleep, wait, join 的线程),会抛出Interr ...
- 关于Java多线程-interrupt()、interrupted()、isInterrupted()解释
多线程先明白一个术语"中断状态",中断状态为true,线程中断. interrupt():就是通知中止线程的,使"中断状态"为true. isInterrupt ...
- JAVA thread0.interrupt()方法
interrupt()只是改变中断状态而已,interrupt()不会中断一个正在运行的线程.这一方法实际上完成的是,给受阻塞的线程抛出一个中断信号,这样受阻线程就得以退出阻塞的状态. 更确切的说,如 ...
随机推荐
- docker在centos7系统镜像下遇到的一些问题
一.成功安装服务后发现无法启动 报错为:Failed to get D-Bus connection: Operation not permitted 系统为centos7官方版镜像,源和依赖之类的都 ...
- 005_awk案例实战
一.工作经验总结. (1)日志案例: 10.100.194.39 10.100.194.39 1019-03-16T11:01:04+08:00 www.uuwatch.com^^3FF91DE01B ...
- jumpserver会话终端故障
查看jumpserver的终端,发现无法访问,情况如下图 页面中提到,需要部署luna,coco和nginx的url分发. 然后进入到jumpserver的服务器,发现nginx启动了,但是没有coc ...
- openvpn用户管理、linux客户端配置及企业常用真实案例解析
1.给企业用户分配VPN账户的流程: 添加拨号需要密码的用户 # source vars NOTE: If you run ./clean-all, I will be doing a rm -rf ...
- Android中PopupWindow用法
参考资料链接:http://developer.android.com/reference/android/widget/PopupWindow.html 在Android中有很多级别的Window, ...
- 一台电脑,两个及多个git账号配置
1. 生成两[三]个ssh公钥私钥 方法参照:http://www.cnblogs.com/fanbi/p/7772812.html第三步骤 假定其中一个是id_rsa, 另一个时id_rsa_two ...
- linux压缩与解压
1.tar -zcvf /home/aaa.tar.gz /xahot tar -zcvf 打包后生成的文件名全路径 要打包的目录 解压 #tar -zxvf /usr/local/test.tar. ...
- BN(Batch Normalization)
Batch Nornalization Question? 1.是什么? 2.有什么用? 3.怎么用? paper:<Batch Normalization: Accelerating Deep ...
- poj3468
#include<iostream> #include<cstring> #include<cstdio> using namespace std; #define ...
- vue组件库(五):icon管理
图标管理 前言 一.常用图标库网站 1.http://icomoon.io 阿里巴巴矢量图 身边的 2.http://fontawesome.dashgame.com iconmoon 3.http: ...