文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java中出现线程中断的原因有哪些

2023-06-06 11:07

关注

这篇文章将为大家详细讲解有关Java中出现线程中断的原因有哪些,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。

一、前言

大家肯定都使用过 Java 线程开发(Thread / Runnable),启动一个线程的做法通常是:

new Thread(new Runnable( @Override public void run() {  // todo sth... })).start();

然而线程退出,大家是如何做的呢?一般做法可能不外乎以下两种:

可能还会有人提出,我可以用中断来退出线程! 我只能说:Too Young Too Simple!中断并不会使得线程结束而退出,中断(interrupt)只是唤醒被阻塞的线程而已。

本篇,我们就来好好的聊聊:线程中断,以及如何正确的使用线程中断,和正确的线程退出。

二、为何 Thread.stop 被废弃

This method is inherently unsafe. Stopping a thread with Thread.stop causes it to unlock all of the monitors that it has locked (as a natural consequence of the unchecked ThreadDeath exception propagating up the stack). If any of the objects previously protected by these monitors were in an inconsistent state, the damaged objects become visible to other threads, potentially resulting in arbitrary behavior. Many uses of stop should be replaced by code that simply modifies some variable to indicate that the target thread should stop running. The target thread should check this variable regularly, and return from its run method in an orderly fashion if the variable indicates that it is to stop running. If the target thread waits for long periods (on a condition variable, for example), the interrupt method should be used to interrupt the wait.

以上是官方 JDK 中的源码注释说明,其含义如下:

**Thread.stop 方法天生就不安全。**使用该方法来停止线程,将会导致其它因为监视器锁『监视器我们在 synchronized 中就讲过,是 Java 的内置锁』而被锁住的线程全部都解锁!(本质的后果是:没有检查的 ThreadDeath 异常会在栈中传播,因而使得监视器锁解锁)。如果任何一个被监视器锁给锁住的对象处于一个不一致的状态,那么其被解锁后将会被其它线程可见,潜在的结果是产生任何后果。**我们应该使用一个变量来代替使用 stop 方法,告诉目标线程退出『这里就是我们开头所说的第一种方法,设置一个标志位』。**目标线程应该周期性的检查这个变量,并根据这个变量来正确的退出 run 方法。如果目标线程处于阻塞/休眠状态(如:使用 wait、sleep、yield 方法后,线程让出了 CPU 使用权,进而阻塞/休眠),此时,该标志位变量将不会起作用,那么,应该使用 interrupt 方法来中断目标线程的阻塞/休眠状态,将其唤醒!

对于 ThreadDeath 对象,官方还有补充:

所以,我们也别想着去 try-catch ThreadDeath Exception!

同样,被废弃的还有 Thread.resume 和 Thread.suspend。这俩方法有造成死锁的危险:

取代这两方法的正确方式是:Object.wait 和 Object.notify :

因为 Object.wait 进入阻塞时,会释放锁。

三、线程中断的含义

Thread 中有三个与中断相关的方法:

注:如果线程中断后,连续两次调用 Thread.interrupted(),第一次是 true & 清除状态,第二次结果是 false。

3.1、初步了解

我们先来通过一个例子来初步了解 thread.interrupt :

public class InterruptDemo implements Runnable { @Override public void run() {  while (true) {   System.out.println("Thread running...");  } } public static void main(String[] args) throws InterruptedException {  Thread thread = new Thread(new InterruptDemo(), "InterruptDemo");  System.out.println("start thread");  thread.start();  Thread.sleep(50);  System.out.println("interrupt thread");  thread.interrupt();  Thread.sleep(50);  System.out.println("thread's status = " + thread.isInterrupted()); }}

输出结果:

start threadThread running...Thread running.........interrupt threadThread running...Thread running.........thread's status = trueThread running.........

我们可以看到,即便我们调用了 thread.interrupt 方法,线程也并没有退出,仍旧继续运行。因此,这个例子证明了一点:我们并不能通过"我们所认为的"中断来试图"结束"正在运行的线程。

3.2、中断即唤醒阻塞/休眠的线程

同样,我们再来看一个例子:

public class InterruptDemo implements Runnable { @Override public void run() {  while (true) {   System.out.println("Thread will sleep 10s ------------------------- running");   long timestamp = System.currentTimeMillis();   try {    Thread.sleep(10000);   } catch (InterruptedException e) {    System.out.println("thread interrupted...");   }   timestamp = System.currentTimeMillis() - timestamp;   System.out.println("Thread run, total sleep = " + timestamp + "(ms)");  } } public static void main(String[] args) throws InterruptedException {  Thread thread = new Thread(new InterruptDemo(), "InterruptDemo");  System.out.println("start thread");  thread.start();  Thread.sleep(3000);  System.out.println("interrupt thread");  thread.interrupt();  System.out.println("main exit"); }}

输出结果:

start threadThread will sleep 10s ------------------------- runninginterrupt threadmain exitthread interrupted...Thread run, total sleep = 3002(ms)Thread will sleep 10s ------------------------- runningThread run, total sleep = 10002(ms)Thread will sleep 10s ------------------------- running

我们可以看到,线程启动后,进入睡眠(10s),3秒后被中断唤醒,执行完一个 while 后再次进入第二次睡眠(10s),然后周而复始。

3.3、一般标志位法退出线程

public class InterruptDemo implements Runnable { private static final AtomicBoolean running = new AtomicBoolean(true); @Override public void run() {  while (running.get()) {   long timestamp = System.currentTimeMillis();   timestamp = System.currentTimeMillis() - timestamp;   System.out.println("Thread run, total sleep = " + timestamp + "(ms)");  }  System.out.println("Thread exit"); } public static void main(String[] args) throws InterruptedException {  Thread thread = new Thread(new InterruptDemo(), "InterruptDemo");  System.out.println("start thread");  thread.start();  Thread.sleep(100);  System.out.println("interrupt thread");  thread.interrupt();  running.set(false);  System.out.println("main exit"); }}

输出结果:

start thread.......Thread run, total sleep = 0(ms)interrupt threadThread run, total sleep = 0(ms)Thread run, total sleep = 0(ms)Thread run, total sleep = 0(ms)main exitThread exit

我们通过使用一个 AtomicBoolean 变量来当作标志位,使得我们的线程能正常退出。 我们也可以判断线程是否被中断而选择性的退出。

3.4、线程中断退出

public class InterruptDemo implements Runnable { @Override public void run() {  while (!Thread.currentThread().isInterrupted()) {   long timestamp = System.currentTimeMillis();   timestamp = System.currentTimeMillis() - timestamp;   System.out.println("Thread run, total sleep = " + timestamp + "(ms)");  }  System.out.println("Thread exit"); } public static void main(String[] args) throws InterruptedException {  Thread thread = new Thread(new InterruptDemo(), "InterruptDemo");  System.out.println("start thread");  thread.start();  Thread.sleep(100);  System.out.println("interrupt thread");  thread.interrupt();  System.out.println("main exit"); }}

输出结果:

start thread.......Thread run, total sleep = 0(ms)interrupt threadThread run, total sleep = 0(ms)Thread run, total sleep = 0(ms)Thread run, total sleep = 0(ms)main exitThread exit

3.5、标志位 + 线程中断结合

public class InterruptDemo implements Runnable { private static final AtomicBoolean running = new AtomicBoolean(true); @Override public void run() {  while (running.get()) {   System.out.println("Thread will sleep 10s ------------------------- running");   long timestamp = System.currentTimeMillis();   try {    Thread.sleep(10000);   } catch (InterruptedException e) {    System.out.println("Interrupted... Todo other things then exit......");    running.set(false);    continue;   }   timestamp = System.currentTimeMillis() - timestamp;   System.out.println("Thread run, total sleep = " + timestamp + "(ms)");  }  System.out.println("Thread exit"); } public static void main(String[] args) throws InterruptedException {  Thread thread = new Thread(new InterruptDemo(), "InterruptDemo");  System.out.println("start thread");  thread.start();  Thread.sleep(3000);  System.out.println("interrupt thread");  thread.interrupt();  System.out.println("main exit"); }}

输出结果:

start threadThread will sleep 10s ------------------------- runninginterrupt threadmain exitInterrupted... Todo other things then exit......Thread exit

关于Java中出现线程中断的原因有哪些就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯