java线程中断

【README】

本文po出了不同场景下线程中断的不同开发方式,包括阻塞,非阻塞,io阻塞线程等;

本文部分内容转自:这篇博文写的非常好

Thread的中断机制(interrupt) - 寂静沙滩 - 博客园先看收集了别人的文章,全面的了解下java的中断:中断线程线程的thread.interrupt()方法是中断线程,将会设置该线程的中断状态位,即设置为true,中断的结果线程是死亡、还是等待新的任务https://www.cnblogs.com/onlywujun/p/3565082.html


【1】使用thread.interrupt()中断【非】阻塞状态线程

子线程不调用 sleep,wait 等方法就不会进入阻塞状态;

本文使用 while() 循环模拟 sleep睡眠方法, 如 tag1代码;

public class Example02 extends Thread {public static void main(String[] args) throws InterruptedException {Example02 thread = new Example02();System.out.println("【主线程】starting thread");thread.start();TimeUnit.SECONDS.sleep(1);// 发出中断请求thread.interrupt(); // tag0  System.out.println("【主线程】发出中断请求后");TimeUnit.SECONDS.sleep(3);System.out.println("【主线程】随眠3秒后");System.out.println("【主线程】stoping thread");}@Overridepublic void run() {while(!Thread.currentThread().isInterrupted()) {  // tag2 long start = System.currentTimeMillis();while(System.currentTimeMillis() - start < 500); // tag1 System.out.println("【子线程】running");}System.out.println("【子线程】current thread exit.");}
}

【代码解说】

  • tag0代码,调用 interrupt 之后,子线程的中断状态会设置为true;
  • 只要不调用阻塞方法,如 sleep,wait 等方法就不会抛出中断异常;(即调用 thread.interrupt方法 是不会让线程中断的,仅仅是设置了中断状态而已)
  • 但tag2判断为false(因为中断状态为true),所以子线程结束循环,正常退出;

【2】使用thread.interrupt()中断阻塞状态线程

线程调用 sleep, wait 方法都可以进入阻塞状态; 

/*** @Description 使用 thread.interrupt() 中断阻塞状态线程,捕获 InterruptedException 异常* @author xiao tang* @version 1.0.0* @createTime 2022年02月20日*/
public class Example02_2 extends Thread {public static void main(String[] args) throws InterruptedException {Example02_2 thread = new Example02_2();System.out.println("【主线程】starting thread");thread.start();TimeUnit.SECONDS.sleep(3);// 发出中断请求thread.interrupt(); // tag0 System.out.println("【主线程】发出中断请求后");TimeUnit.SECONDS.sleep(3);System.out.println("【主线程】随眠3秒后");System.out.println("【主线程】stoping thread");}@Overridepublic void run() {while(!Thread.currentThread().isInterrupted()) { // tag1 try {TimeUnit.MILLISECONDS.sleep(500); // 可中断式阻塞 tag2} catch (InterruptedException e) { // tag3 System.out.println("抛出中断异常,线程中断状态=" + Thread.currentThread().isInterrupted()); // false }System.out.println("【子线程】running");}System.out.println("【子线程】current thread exit.");}
}

【 代码解说】

情况1: 主线程tag0代码执行后, 设置线程中断状态为true;子线程立马执行tag1,循环结束,子线程退出;(子线程被中断,正常退出)

情况2:主线程tag0代码执行后, 设置线程中断状态为true; 子线程 调用Thread.sleep(500) 会抛出中断异常 InterruptedException(因为中断状态为true),而抛出InterruptedException异常后,java底层会把中断标示位会自动清除(中断状态设置false);接着 while循环中的 !Thread.currentThread().isInterrupted() 为判断为true,接着继续执行循环体逻辑;(子线程被中断,无法正常退出)

如何解决情况2中的子线程无法正常退出的情况?

在情况2的catch块中调用  

Thread.currentThread().interrupt();//重新设置中断状态为true 

重新设置中断状态为true ;然后while的判断语句就会判断为false,结束死循环,从而子线程才可以正常退出

当然,如果业务逻辑不需要让子线程结束,即便遇到中断的话,无需添加 Thread.currentThread().interrupt() 代码;


【3】死锁状态线程无法被中断

/*** @Description 死锁状态线程无法被中断* @author xiao tang* @version 1.0.0* @createTime 2022年02月20日*/
public class Example03_DeadLock {static class MyLock {String name;MyLock(String name) {this.name = name;}}public static void main(String[] args) {final MyLock lock1 = new MyLock("锁1");final MyLock lock2 = new MyLock("锁2");// 线程1Thread t1 = new Thread(new Runnable() {public void run() {lock(lock1, lock2); // 传入锁的顺序不一样 }}, "线程1");// 线程2Thread t2 = new Thread(new Runnable() {public void run() {lock(lock2, lock1); // 传入锁的顺序不一样}}, "线程2");// 开启线程t1.start();t2.start();// 中断线程t1.interrupt(); // 设置线程1的中断状态为truet2.interrupt(); // 设置线程2的中断状态为trueSystem.out.println("【主线程】执行完成");}static void lock(MyLock lock1, MyLock lock2) {synchronized (lock1) { // 获得锁1System.out.println(Thread.currentThread().getName() + "获得" + lock1.name);// 获取锁2前先睡眠3秒 以便线程2获取锁1,才能模拟出死锁try {TimeUnit.SECONDS.sleep(3);} catch (InterruptedException e) {}synchronized (lock2) { // 获得锁2System.out.println(Thread.currentThread().getName() + "获得" + lock2.name);}}}
}

线程1与线程2阻塞,程序无法正常结束;


【4】中断IO阻塞的线程

/*** @Description 中断IO阻塞线程* @author xiao tang* @version 1.0.0* @createTime 2022年02月20日*/
public class Eaxmple04_IOBlock extends Thread {volatile ServerSocket serverSocket; // 服务器套接字public static void main(String[] args) throws Exception {Eaxmple04_IOBlock ioBlockThread = new Eaxmple04_IOBlock();ioBlockThread.start();// 主线程睡眠等待子线程打开套接字TimeUnit.SECONDS.sleep(3);// 中断子线程ioBlockThread.interrupt(); // 中断 tat0
//        ioBlockThread.serverSocket.close(); // tag3System.out.println("【主线程】关闭了服务器套接字");System.out.println("【主线程】结束");}@Overridepublic void run() {try {serverSocket = new ServerSocket(8080);} catch (IOException e) {System.out.println("【子线程】打开套接字失败,直接返回");return ;}while(!Thread.currentThread().isInterrupted()) { // tag2try {System.out.println("【子线程】接收客户端请求");serverSocket.accept();} catch (IOException e) { // 抛出IO异常后, java底层会把当前线程中断状态重置为false(清空)System.out.println("【子线程】accept()遇到io异常");
//                Thread.currentThread().interrupt(); // 重新设置当前线程中断状态为true  tag1} finally {try {serverSocket.close();System.out.println("【子线程】服务器套接字关闭");} catch (IOException e) {System.out.println("【子线程】服务器套接字关闭出现异常");}}}System.out.println("【子线程】io thread结束");}
}

【注】注释了 tag3  tag1 的代码;

以上代码的子线程无法正常结束;

服务器套接字 ServerSocket.accept() 没有抛出 InterruptedException 中断异常,所以它是非中断式阻塞;所以 accept() 一直阻塞(子线程也一直阻塞),无法中断;

解决方法:

添加 tag3代码,主线程关闭 服务器套接字;

ioBlockThread.serverSocket.close(); // tag3

且 添加 tag1代码 重新设置当前线程中断状态为true;如下:

Thread.currentThread().interrupt(); // 重新设置当前线程中断状态为true  tag1

因为抛出 IOException状态后,需要重新把当前线程中断状态设置为true,以便退出while循环,从而子线程才会正常结束;

注意:tag0的代码  ioBlockThread.interrupt() 不能删除,一旦删除, 子线程将无法正常退出,可自行实验

【4.1】如何中断IO操作 (使用可中断通道)

中断I/O操作

然而,如果线程在I/O操作进行时被阻塞,又会如何?I/O操作可以阻塞线程一段相当长的时间,特别是牵扯到网络应用时。例如,服务器可能需要等待一个请求(request),又或者,一个网络应用程序可能要等待远端主机的响应。

实现此InterruptibleChannel接口的通道是可中断的:如果某个线程在可中断通道上因调用某个阻塞的 I/O 操作(常见的操作一般有这些:serverSocketChannel. accept()、socketChannel.connect、socketChannel.open、socketChannel.read、socketChannel.write、fileChannel.read、fileChannel.write)而进入阻塞状态,而另一个线程又调用了该阻塞线程的 interrupt 方法,这将导致该通道被关闭,并且已阻塞线程接将会收到ClosedByInterruptException,并且设置已阻塞线程的中断状态。另外,如果已设置某个线程的中断状态并且它在通道上调用某个阻塞的 I/O 操作,则该通道将关闭并且该线程立即接收到 ClosedByInterruptException;并仍然设置其中断状态。如果情况是这样,其代码的逻辑和第三个例子中的是一样的,只是异常不同而已。

如果你正使用通道(channels)(这是在Java 1.4中引入的新的I/O API),那么被阻塞的线程将收到一个ClosedByInterruptException异常。但是,如果你使用Java1.0之前就存在的传统的I/O,而且要求更多的工作。既然这样,Thread.interrupt()将不起作用,因为线程将不会退出被阻塞状态。尽管interrupt()被调用,线程也不会退出被阻塞状态,比如ServerSocket的accept方法根本不抛出异常。

很幸运,Java平台为这种情形提供了一项解决方案,即调用阻塞该线程的套接字的close()方法。在这种情形下,如果线程被I/O操作阻塞,当调用该套接字的close方法时,该线程在调用accept地方法将接收到一个SocketException(SocketException为IOException的子异常)异常,这与使用interrupt()方法引起一个InterruptedException异常被抛出非常相似,(注,如果是流因读写阻塞后,调用流的close方法也会被阻塞,根本不能调用,更不会抛IOExcepiton,此种情况下怎样中断?我想可以转换为通道来操作流可以解决,比如文件通道)。下面是具体实现: 如  Eaxmple04_IOBlock


【5】小结

  • 一、没有任何语言定义了一个被中断的线程应该终止。中断一个线程只是为了引起该线程的注意,被中断线程可以决定如何应对中断。
  • 二、对于处于sleep,join等操作的线程,如果被调用interrupt()后,会抛出InterruptedException,然后线程的中断标志位会由true重置为false(只要抛出中断异常,则中断标识修改为false,即清空中断标识),因为线程为了处理异常已经重新处于就绪状态。
  • 三、不可中断的操作,包括进入synchronized段以及Lock.lock(),inputSteam.read()等,调用interrupt()对于这几个问题无效,因为它们都不抛出中断异常。如果拿不到资源,它们会无限期阻塞下去。
  • 四:可中断操作:对于Lock.lock(),可以改用Lock.lockInterruptibly(),可被中断的加锁操作,它可以抛出中断异常。等同于等待时间无限长的Lock.tryLock(long time, TimeUnit unit)。对于inputStream等资源,有些(实现了 InterruptibleChannel 接口)可以通过close()方法将资源关闭,对应的阻塞也会被放开。

【5.1】Thread类里的几个方法:

  1. public static boolean interrupted: 测试当前线程是否已经中断。线程的中断状态 由该方法清除。换句话说,如果连续两次调用该方法,则第二次调用将返回 false。
  2. public boolean isInterrupted():  测试线程是否已经中断。线程的中断状态 不受该方法的影响。
  3. public void interrupt(): 中断线程。

上面列出了与中断有关的几个方法及其行为,可以看到interrupt是中断线程。如果不了解Java的中断机制,这样的一种解释极容易造成误解,认为调用了线程的interrupt方法就一定会中断线程

【5.2】其实,Java的中断是一种协作机制。

也就是说调用线程对象的interrupt方法并不一定就中断了正在运行的线程,它只是要求线程自己在合适的时机中断自己。每个线程都有一个boolean的中断状态(这个状态不在Thread的属性上),interrupt方法仅仅只是将该状态置为true。比如对正常运行的线程调用interrupt()并不能终止他,只是改变了interrupt标示符。

【5.3】如何判断一个方法是否可中断

一般说来,如果一个方法声明抛出InterruptedException,表示该方法是可中断的,比如wait,sleep,join,也就是说可中断方法会对interrupt调用做出响应(例如sleep响应interrupt的操作包括清除中断状态,抛出InterruptedException),异常都是由可中断方法自己抛出来的,并不是直接由interrupt方法直接引起的。

Object.wait, Thread.sleep方法,会不断的轮询监听 interrupted 标志位,发现其设置为true后,会停止阻塞并抛出 InterruptedException异常。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/329697.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WEB API系列(一):WEB API的适用场景、第一个实例

在我前一篇博客《WebAPI前置知识:HTTP与RestfulAPI》中已经给各位简单介绍了HTTP协议与RestFul API的关系&#xff0c;以及一些基本的HTTP协议知识&#xff0c;在这些知识的铺垫下&#xff0c;今天&#xff0c;我们一起来讨论一下WEB API的适用场景&#xff0c;然后写我们第一个…

rpa文件怎么提取内容_怎么编辑pdf文件内容?有什么软件可以编辑pdf文件吗?

怎么编辑pdf文件内容&#xff1f;我是2020年大学新生&#xff0c;选择学习的专业是财务管理。为了更快、更深入地了解更多专业知识&#xff0c;我上网查找、下载了很多相关资料。不过&#xff0c;从网上保存下来的资料文件大多是pdf格式的&#xff0c;想要编辑它还得先转word。…

Future取消线程执行

【README】 本文总结于 《java并发编程实战》 page121&#xff0c;非常棒的一本书&#xff1b; 【1】Future 1&#xff0c;介绍&#xff1a;future 用于管理任务的生命周期&#xff0c;处理异常&#xff0c;以及实现取消&#xff1b; 2&#xff0c;future.cancel() 取消方法…

Nancy之结合tinyfox给我们的应用提供简单的数据服务

说到提供数据服务给我们的一些应用&#xff0c;估计用的最多的也就是json和xml这两种数据格式 实现的方法也是多种多样&#xff0c;web api,mvc的jsonresult和contentresult...等等 本文是结合Nancy、TInyFox、Owin等来实现的 一、前提工作 新建一个空的web应用程序 添加相应的…

javaweb调用python算法_请教怎么用java远程调用python? 多谢

请问如何用java远程调用python? 谢谢&#xff01;本帖最后由 blackkettle 于 2015-05-07 13:00:41 编辑比如有一台机器 A上安装了python, 另一台机器B要用java 调用A 上的python的某个函数&#xff0c;输入数据在B机器上&#xff0c;所有的计算在A机器完成&#xff0c;结果返回…

(转)如何查看java本地方法

在线查看本地代码&#xff0c; refer2 http://hg.openjdk.java.net/jdk8/jdk8/hotspot/file/tip/src/os/linux/vm/os_linux.cpp 我们知道在java中查看java源码时看到native方法在java层面上就到头的&#xff0c;如果还想继续往下看就需要看jdk是如何实现的&#xff0c;今天就分…

统一配置中心的设计方案

转载自 统一配置中心的设计方案对于配置文件&#xff0c;我们不陌生&#xff0c;它提供我们可以动态修改程序运行能力。引用别人的一句话就是&#xff1a;系统运行时(runtime)飞行姿态的动态调整。我可以把我们的工作称之为在快速飞行的飞机上修理零件。我们人类总是无法掌控和…

微软CEO纳德拉恢弘计划:让开发者始终忘不了微软

BI中文站 4月11日 报道 当微软前CEO史蒂夫•鲍尔默&#xff08;Steve Ballmer&#xff09;在2000年喊出“开发者&#xff0c;开发者&#xff0c;开发者”的口号时&#xff0c;他可能有点儿激动&#xff0c;但是他的观点是对的。 30年前当Windows 1.0推出的时候&#xff0c;这种…

python scratch unity_Unity3D研究院之2D游戏开发制作原理(二十一)

经过了4个月不懈的努力&#xff0c;我和图灵教育合作的这本3D游戏开发书预计下个月就要出版了。这里MOMO先打一下广告&#xff0c;图灵的出版社编辑成员都非常给力&#xff0c;尤其是编辑小花为这本书付出了很大的努力&#xff0c;还有杨海玲老师&#xff0c;不然我也无法完成这…

原码补码与反码

【README】 1.本文内容总结自“哈工大刘宏伟”老师的mooc视频《计算机组成原理》on bilibili&#xff1b; 2.为了便于理解&#xff0c;本文引入了逗号分割符号部分和数值部分&#xff0c;计算机存储数据的时候没有逗号&#xff1b; 【1】原码表示法 原码定义&#xff1a; 原…

第一篇 Entity Framework Plus 之 Audit

一般系统会有登陆日志&#xff0c;操作日志&#xff0c;异常日志&#xff0c;已经满足大部分的需求了。但是有时候&#xff0c;还是需要Audit 审计日志&#xff0c;审计日志&#xff0c;主要针对数据增&#xff0c;改&#xff0c;删操作数据变化的记录&#xff0c;主要是对数据…

本想试试看,结果却拿到了京东的Offer

转载自 本想试试看&#xff0c;结果却拿到了京东的Offer 最近&#xff0c;春招已经基本接近尾声了&#xff0c;我找了几位拿到名企Offer的粉丝&#xff0c;请他们总结了面试经验&#xff0c;近期会分批的推送给大家。希望给那些正在准备秋招的同学提供些帮助。 今天给大家分享的…

ping 命令使用代理_网络检测知识篇:ping命令使用知识,你知道几点?

Ping命令其实是一个非常好的网络故障诊断工具&#xff0c;相信阅读完本文对大家一定有些帮助。如果大家网络遇到问题&#xff0c;不妨试试以下方法。首先使用Ping命令诊断本地TCP/IP协议是否安装正常&#xff0c;检测方法如下&#xff1a;从电脑开始里找到运行&#xff0c;快捷…

8.4-中断系统小结(cpu中断七个问题)

【README】 本文转自bilibili《计算机组成原理&#xff08;哈工大刘宏伟&#xff09;》的视频讲解&#xff0c;非常棒&#xff0c;墙裂推荐&#xff1b; 【1】中断介绍 1&#xff09;作用&#xff1a;用中断系统实现了外设数据的输入输出&#xff1b; 还可以用于程序调试&…

第二篇 Entity Framework Plus 之 Query Future

从性能的角度出发&#xff0c;能够减少 增&#xff0c;删&#xff0c;改&#xff0c;查&#xff0c;跟数据库打交道次数&#xff0c;肯定是对性能会有所提升的&#xff08;这里单纯是数据库部分&#xff09;。 今天主要怎样减少Entity Framework查询跟数据库打交道的次数&#…

python长整数相乘_python写的大整数相乘的方法

输入72106547548473106236 982161082972751393 两个大整数输出结果70820244829634538040848656466105986748解题思路首先根据 大整数相乘的原理的基础上&#xff0c;把大整数进行优化拆分&#xff0c;拆分的长度&#xff0c;要考虑语言中整形的长度。这里用的python&#xff0c…

Java Web应用的代码分层最佳实践

转载自 Java Web应用的代码分层最佳实践代码分层&#xff0c;对于任何一个Java Web开发来说应该都不陌生。一个好的层次划分不仅可以能使代码结构更加清楚&#xff0c;还可以使项目分工更加明确&#xff0c;可读性大大提升&#xff0c;更加有利于后期的维护和升级。从另外一个角…

中断屏蔽技术

【README】 本文总结自bilibili《计算机组成原理&#xff08;哈工大刘宏伟&#xff09;》的视频讲解&#xff0c;非常棒&#xff0c;墙裂推荐&#xff1b; 【1】中断屏蔽 1&#xff0c;中断屏蔽的意思是&#xff0c;在中断1的服务程序执行过程中&#xff0c;不允许被其他中断…

.NET FM的未来计划

2016年3月21日&#xff0c;我们启动了.NET FM这档独立播客来服务中文.NET和微软技术社区。如同早先感谢信所言&#xff0c;能够得到社区的肯定和全力支持&#xff0c;二位主播是受宠若惊。关于未来的计划&#xff0c;下面简单和大家分享一下。 首先&#xff0c;关于节目播出频率…

Java开发必须掌握的日志分析命令

转载自 Java开发必须掌握的日志分析命令 对于大型网站来说&#xff0c;很多网站在可用性方面提出4个9或者5个9的要求&#xff0c;如果是4个9&#xff0c;那么网站全年的不可用时间不能超过52.6分钟&#xff0c;如果是5个9&#xff0c;全年不可用时间不能超过5.2分钟。这其实是很…