Java线程中断interrupt的常用方法
前言
这里主要探讨中断常用的三个方法:
- interrupt()。在一个线程中调用需要中断现成的interrupt()方法,会对该线程发出信号,将中断状态标志为true
- isInterrupted()。判断当前线程的中断状态。
- interrupted()。将线程的中断状态恢复。
主要使用的阻塞三个方法:
- Object#wait。放弃锁+等待+重新获取锁
- Thread#join。【协作】等待某个线程执行完毕
- Thread#sleep。静态方法,线程休眠并让出CPU时间片
==注意:interrupt()不能中断在运行中的线程,它只能改变中断状态而已。实际完成的是让受阻塞的线程退出阻塞状态。==
确切的说:是被三种方法之一阻塞时,调用该线程的interrupt()方法,那么线程将抛出一个个InterruptedException中断异常,从而提早地终结被阻塞状态。
示例说明
public class Runner3 implements Runnable { @Override public void run() { while (true) { if (Thread.currentThread().isInterrupted()) { System.out.println("我进入中断了,但我还在跑"); } else { System.out.println("我没有进入中断"); } } } public static void main(String[] args) { Runner3 runner3 = new Runner3(); Thread thread3 = new Thread(runner3); thread3.start(); try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } thread3.interrupt(); } }
输出结果大致如下:
我没有进入中断
我没有进入中断
我进入中断了,但我还在跑
我进入中断了,但我还在跑
我进入中断了,但我还在跑
...
这里看到,执行interrupt()后,对线程执行中断后依然在执行,线程依然在运行。
我们调整一下run方法
public void run() { while (true) { if (Thread.currentThread().isInterrupted()) { System.out.println("我进入中断了,但我还在跑"); Thread.interrupted();//重置状态 } else { System.out.println("我没有进入中断"); } } }
输出结果如下:
我没有进入中断
我没有进入中断
我进入中断了,但我还在跑
我没有进入中断
我没有进入中断
...
这里看到中断的状态重置了,那么我们如何去应用这个中断状态呢?
注意事项
- 当线程A执行到wait(),sleep(),join()时,抛出InterruptedException后,中断状态已经被系统复位了,线程A调用Thread.interrupted()返回的是false。
- 如果线程被调用了interrupt(),此时该线程并不在阻塞状态时,下次执行wait(),sleep(),join()时,一样会抛出InterruptedException,当然抛出后该线程的中断状态也会被系统复位。
案例1
public class Runner3 implements Runnable { @Override public void run() { while (true) { if (Thread.currentThread().isInterrupted()) { System.out.println("我进入中断了,但我还在跑"); // try { Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); System.out.println("2"+Thread.currentThread().isInterrupted()); //输出false } } else { System.out.println("我没有进入中断"); } } } public static void main(String[] args) { Runner3 runner3 = new Runner3(); Thread thread3 = new Thread(runner3); thread3.start(); try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } thread3.interrupt(); } }
执行上面的代码,我们可以看到在抛出异常后,Thread.currentThread().isInterrupted()输出为false,证明线程的中断状态已经复位了。
另外因为我们是先执行了interrupt()然后再进入睡眠状态,但是依然抛出了异常。
Object#wait 和 Thread.sleep 差异在哪里
因为Object#wait方法会阻塞线程,所以当我们执行interrupt时,会抛出InterruptedException异常。
那么Object#wait方法阻塞线程会导致的差异在哪里?
==最主要的差别在于sleep方法没有释放锁,而wait方法释放了锁,使得其它线程可以使用同步控制块或者方法。==
总结
- 调用interrupt方法,会改变中断状态,但不会影响线程的运行状态。
- 当执行了interrupt方法改变中断状态后,线程若执行Object#wait,Thread#sleep和Thread#join都会抛出InterruptedException异常,然后复位中断状态
- 当执行了interrupt方法改变中断状态后,线程未阻塞,且将要执行Object#wait,Thread#sleep和Thread#join阻塞线程时,都会抛出InterruptedException异常,复位中断状态。
到此这篇关于Java线程中断interrupt的常用方法的文章就介绍到这了,更多相关Java线程中断interrupt内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
Spring Boot 整合 Fisco Bcos部署、调用区块链合约的案例
本篇文章介绍 Spring Boot 整合 Fisco Bcos 的相关技术,最最重要的技术点,部署、调用区块链合约的工程案例,本文通过流程分析给大家介绍的非常详细,需要的朋友参考下吧2022-01-01从源码角度简单看StringBuilder和StringBuffer的异同(全面解析)
下面小编就为大家分享一篇从源码角度简单看StringBuilder和StringBuffer的异同(全面解析),具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧2017-12-12elasticsearch集群cluster discovery可配式模块示例分析
这篇文章主要为大家介绍了elasticsearch集群cluster discovery可配式模块示例分析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪2022-04-04SpringDataElasticsearch与SpEL表达式实现ES动态索引
这篇文章主要介绍了SpringDataElasticsearch与SpEL表达式实现ES动态索引,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的朋友可以参考一下2022-09-09一文教会Java新手使用Spring MVC中的查询字符串和查询参数
在使用springMVC框架构建web应用,客户端常会请求字符串、整型、json等格式的数据,这篇文章主要给大家介绍了关于通过一文教会Java新手使用Spring MVC中的查询字符串和查询参数的相关资料,需要的朋友可以参考下2024-01-01
最新评论