Java Concurrency/Threads(2)——线程中断

NOTE:参考自blog:https://www.jianshu.com/p/db8f3f0cdf15

1. 线程中断的概念与方法

  1. 中断: 每个线程有一个中断状态,且初始时,中断状态为false。

  2. 当线程A通过调用Thread.interrupt()中断一个线程B,会出现一下两种情况之一:

    a、如果线程B正在执行一个低级可中断阻塞方法,如Thread.sleep(), Thread.join(), Object.wait(),则取消阻塞并且抛出InterruptedException。

    b、相反,interrupt()方法只是设置线程B的中断状态为true, 当线程B询问中断状态时候(Thread.currentThread().isInterrupted()读取中断状态,并且可以通过Thread.interrupted()的操作读取和清除中断位(变为true))。

  3. 因此,当一个线程中断另外一个线程时候,被中断的线程不一定要立即停止正在做的工作

停止线程:在线程处理完任务之前,停掉正在做的工作。

Java中,3种终止正在运行的线程的方法:

  1. 使用退出标志,使线程正常退出,即run方法完成后正常停止。

  2. 使用Thread.stop() 强制终止线程,过期作废的方法,排除不用。

  3. 使用interrupt方法终止线程。

NOTE: 采用Thread.interrupt()方法,这个方法不会直接终止一个正在运行的线程,还需要加入一个判断才能够完成线程的停止。

2. interrupt 方法终止线程的具体操作

1. interrupt + for break

package com.guoqiang;

public class BreakInterrupt extends Thread {
    @Override
    public void run() {
        super.run();
        for (int i = 0; i < 50000; i++) {
            if (Thread.currentThread().isInterrupted()) {    // 用Thread.currentThread().isInterrupted() 方法判断当前线程是否被中断
                System.out.println("要退出了,拜拜~");
                break;  // 使用break退出 for 循环
            }
            System.out.println("i = " + (i + 1));
        }
        System.out.println("如果此段代码在for循环中继续运行,线程就未停止~");
    }
}

public class Main {

    public static void main(String[] args) {
        Thread breakInterrupt = new BreakInterrupt();
        breakInterrupt.start();
        try {
            Thread.sleep(400);
            breakInterrupt.interrupt();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("END");
    }
}

2. 将interrupt和return结合起来 结束线程

if (Thread.currentThread().isInterrupted()) {
        System.out.println("this Thread END");
        return;
}

3. 中断相关函数

  1. Thread.interrupt() :
    (1)将线程的中断状态设置成为true;
    (2)让被阻塞的线程抛出InterruptException异常(同时中断状态为false)
    注意:没有占用CPU运行的线程是不可能给自己的中断状态置位的,就会产生一个InterruptedException异常。

比如说:线程A获得了锁进入了同步代码块中,但由于条件不足调用 wait() 方法阻塞了。这个时候,线程B执行 threadA.interrupt()请求中断线程A,此时线程A就会抛出InterruptedException,我们就可以在catch中捕获到这个异常并进行相应处理(比如进一步往上抛出)

  1. Thread.isInterrupted():
    检测调用该方法的线程是否被中断,中断状态不会被清除。线程一旦被中断,该方法返回true,而一旦sleep等方法抛出异常,它将清除中断状态,此时方法将返回false。

  2. Thread.interrupted() : 静态方法
    检测当前线程是否被中断,并且中断状态会被清除(即重置为false);由于它是静态方法,因此不能在特定的线程上使用,只能报告调用它的线程的中断状态;如果该方法被调用两次,则第二次一般是返回false,如果线程不存活,则返回false。


public class Main {
    public static void main(String[] args) {
        System.out.println("1: " + Thread.interrupted());
        Thread.currentThread().interrupt();
        System.out.println("2: " + Thread.interrupted());
        System.out.println("3: " + Thread.interrupted());
    }
}
OUT:
1: false
2: true
3: false
posted @ 2019-09-05 17:04  guoqiangliu  阅读(188)  评论(0编辑  收藏  举报