美文网首页
java线程中断原理

java线程中断原理

作者: someoneYoung | 来源:发表于2019-01-17 19:40 被阅读59次

稍有java基础的同学都知道,在java中创建并启动一个线程比较容易,而线程中断的难度更高一些,并且使用的场景也相对较少。

interrupt()

中断某一个线程需要调用该线程对象的interrupt方法。

public class Demo {
    public static void main(String[] args) {
        Thread thread = new Thread(new MyTask());
        thread.start();
        try {
            Thread.sleep(1000);
        } catch (Exception ex) {
        }
        thread.interrupt();
    }
    static class MyTask implements Runnable {
        @Override
        public void run() {
            while (true) {
                System.out.println(Thread.currentThread() + " is running...");
            }
        }
    }
}

运行代码会发现,即使在主线程中执行目标线程的interrupt()方法,但目标线程并没有停止执行。这正是interrupt机制设计的特别之处,当主线程发起目标线程中断的命令后,目标线程并不会立即放弃线程的执行权。

中断标志位

java interrupt中断机制是当主线程向目标线程发起interrupt中断命令后,目标线程的中断标志位被置为true,目标线程通过查询中断标志位自行决定是否停止当前线程的执行。
这便解释了上面的代码中,目标线程的中断标志位虽然置为true,但由于并没有主动采取线程停止的操作,所以线程依然处于Running状态。
查询线程中断标志位的方法有两种:isInterrupted()和interrupted(),下面分别介绍二者的区别。

isInterrupted()与interrupted()

public static boolean interrupted() {
    return currentThread().isInterrupted(true);
}

public boolean isInterrupted() {
    return isInterrupted(false);
}

private native boolean isInterrupted(boolean ClearInterrupted);

直接看这两个方法的源码,interrupted()是静态方法而isInterrupted()是实例方法,他们的实现都是调用同一个native方法。主要的区别是他们的形参ClearInterrupted传的不一样。interrupted()在返回中断标志位后会清除标志位,isInterrupted()则不清除中断标志位。
接下来改造前面的代码,实现线程中断效果:

public class Demo {

    public static void main(String[] args) {
        Thread thread = new Thread(new MyTask());
        thread.start();
        try {
            Thread.sleep(100);
        } catch (Exception ex) {
        }
        thread.interrupt();
    }

    static class MyTask implements Runnable {
        @Override
        public void run() {
            while (true) {
                if (Thread.interrupted()) {
                    break;
                }
                System.out.println(Thread.currentThread() + " is running...");
            }
            System.out.println("当前中断标志位状态:" + Thread.currentThread().isInterrupted());
        }
    }
}
image.png

InterruptedException

调用Thread.sleep()时都需要捕获InterruptedException异常。这个异常的作用是什么?
如果目标线程正在执行阻塞方法(sleep、join),而其他线程恰好调用了目标线程的interrupt方法试图中断目标线程,sleep、join这类阻塞方法会检查线程的中断标志位,并抛出InterruptedException异常。

阻塞方法为何抛出InterruptedException

@Override
public void run() {
    while (true) {
        if (Thread.interrupted()) {
            break;
        }
        // point 1 : 阻塞方法前逻辑
        try {
            // point 2 : 阻塞方法中
            Thread.sleep(10*000*000);
        } catch(InterruptedException ex) {
            // 执行清除逻辑
        }
        // point 3 :阻塞方法后逻辑
    }
}

前文提到过,interrupt的线程中断机制是由发起线程将目标线程的中断标志位置为true,至于是否执行线程的中断由目标线程决定。
以上面代码为例,如果目标线程正在执行sleep方法而线程阻塞,必须在10000000时间完成且并执行完后续逻辑,直至循环里下次interrupted()判断后才能中断线程。显然这不是我们希望看到的,所以阻塞方法会判断中断标志位,一旦出现中断的命令就会抛出异常,直接终止阻塞逻辑。

InterruptedException清空中断标志位

抛出InterruptedException异常也会清除中断标志位,如果想要继续保留中断标志位的状态,可以手动触发中断标志,代码如下:

try {
  Thread.sleep(100);
} catch (InterruptedException ex) {
  Thread.currentThread().interrupt(); 
  throw new RuntimeException(ex);
}

总结

要想理解java线程中断的原理,重点就是要掌握中断标志位的使用细节,其他的逻辑都是围绕中断标志位设计。

相关文章

  • java线程中断原理

    稍有java基础的同学都知道,在java中创建并启动一个线程比较容易,而线程中断的难度更高一些,并且使用的场景也相...

  • Java线程中断的原理

    在java中不提供抢占式中断,也就是说不允许线程之间的直接抢占,(已经有过Thread.stop,Thread.s...

  • Java线程中断

    本文主要介绍Java线程中断一些相关的概念以及注意点 Java线程的中断并不是强制的中断,调用线程中断的方法时只是...

  • 线程中断

    什么是线程中断?线程中断即线程运行过程中被其他线程打断了。 线程中断的重要方法2.1 java.lang.Thre...

  • JAVA并发编程(三)线程协作与共享

    1. 线程中断 java线程中断是协作式,而非抢占式 1.1. 线程中断相关方法 interrupt()将线程的中...

  • 【多线程】——3.线程的中断

    线程中断的概念 java线程中断是一种协作机制 通过中断并不能直接终止线程的运行 需要被中断的线程自己处理中断 (...

  • (4)线程中断

    什么是线程中断 java中的线程中断并不是指强制线程中断执行,而是指调用线程中断起到一个通知作用,让线程知道自己被...

  • 「阿里面试系列」面试加分项,从jvm层面了解线程的启动和停止

    线程的启动的实现原理 线程停止的实现原理分析 为什么中断线程会抛出InterruptedException 线程的...

  • 线程中断

    Java的中断是一种协作机制,线程中断不会终止线程的运行,但是可以通过线程中断来实现终止线程运行。 线程在不同状态...

  • Java “优雅”地中断线程(原理篇)

    前言 线程中断系列文章: Java “优雅”地中断线程(实践篇)[https://www.jianshu.com/...

网友评论

      本文标题:java线程中断原理

      本文链接:https://www.haomeiwen.com/subject/xdtwdqtx.html