一文搞懂 Java 線程中斷

在以前的一文《如何"優雅"地終止一個線程》中詳細說明了 stop 終止線程的壞處及如何優雅地終止線程,那麼還有別的能夠終止線程的方法嗎?答案是確定的,它就是咱們今天要分享的——線程中斷。java

下面的這斷代碼你們應該再熟悉不過了,線程休眠須要捕獲或者拋出線程中斷異常,也就是你在睡覺的時候忽然有我的衝進來把你吵醒了。面試

try {
    Thread.sleep(3000);
} catch (InterruptedException e) {
    e.printStackTrace();
}

此時線程被打斷後,代碼會繼續運行或者拋出異常結束運行,這並非咱們須要的中斷線程的做用。微信

究竟是什麼是線程中斷?

線程中斷即線程運行過程當中被其餘線程給打斷了,它與 stop 最大的區別是:stop 是由系統強制終止線程,而線程中斷則是給目標線程發送一箇中斷信號,若是目標線程沒有接收線程中斷的信號並結束線程,線程則不會終止,具體是否退出或者執行其餘邏輯由目標線程決定。多線程

咱們來看下線程中斷最重要的 3 個方法,它們都是來自 Thread 類!架構

一、java.lang.Thread#interrupt分佈式

中斷目標線程,給目標線程發一箇中斷信號,線程被打上中斷標記。spa

二、java.lang.Thread#isInterrupted()線程

判斷目標線程是否被中斷,不會清除中斷標記。架構設計

三、java.lang.Thread#interrupted設計

判斷目標線程是否被中斷,會清除中斷標記。

線程中斷實戰

咱們來實例演示下線程中斷如何用!

示例1(中斷失敗)

/**
 * 微信公衆號:Java技術棧
 */
private static void test1() {
    Thread thread = new Thread(() -> {
        while (true) {
            Thread.yield();
        }
    });
    thread.start();
    thread.interrupt();
}

請問示例1中的線程會被中斷嗎?答案:不會,由於雖然給線程發出了中斷信號,但程序中並無響應中斷信號的邏輯,因此程序不會有任何反應。

示例2:(中斷成功)

/**
 * 微信公衆號:Java技術棧
 */
private static void test2() {
    Thread thread = new Thread(() -> {
        while (true) {
            Thread.yield();

            // 響應中斷
            if (Thread.currentThread().isInterrupted()) {
                System.out.println("Java技術棧線程被中斷,程序退出。");
                return;
            }
        }
    });
    thread.start();
    thread.interrupt();
}

咱們給示例2加上了響應中斷的邏輯,程序接收到中斷信號打印出信息後返回退出。

示例3(中斷失敗)

/**
 * 微信公衆號:Java技術棧
 */
private static void test3() throws InterruptedException {
    Thread thread = new Thread(() -> {
        while (true) {
            // 響應中斷
            if (Thread.currentThread().isInterrupted()) {
                System.out.println("Java技術棧線程被中斷,程序退出。");
                return;
            }

            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                System.out.println("Java技術棧線程休眠被中斷,程序退出。");
            }
        }
    });
    thread.start();
    Thread.sleep(2000);
    thread.interrupt();
}

示例3 sleep() 方法被中斷,並輸出了 Java技術棧線程休眠被中斷,程序退出。 程序繼續運行……爲何呢?

來看 sleep 的源碼:

image

能夠看出 sleep() 方法被中斷後會清除中斷標記,因此循環會繼續運行。。

示例4(中斷成功)

/**
 * 微信公衆號:Java技術棧
 */
private static void test4() throws InterruptedException {
    Thread thread = new Thread(() -> {
        while (true) {
            // 響應中斷
            if (Thread.currentThread().isInterrupted()) {
                System.out.println("Java技術棧線程被中斷,程序退出。");
                return;
            }

            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                System.out.println("Java技術棧線程休眠被中斷,程序退出。");
                Thread.currentThread().interrupt();
            }
        }
    });
    thread.start();
    Thread.sleep(2000);
    thread.interrupt();
}

示例4所有信息輸出並正常退出,只是在 sleep() 方法被中斷並清除標記後手動從新中斷當前線程,而後程序接收中斷信號返回退出。

經過以上 4 箇中斷示例,相信對 Java 線程中斷的概念有了全面的瞭解。更多 Java 多線程技術文章請在後臺回覆關鍵字:多線程。


更多幹貨推薦

1.史上最強 Java 中高級面試題整理

2.史上最強 Spring Boot & Cloud 教程整理

3.史上最強架構設計分佈式技術乾貨整理

更多請掃描下方的二維碼關注咱們的微信公衆號,乾貨每日推送!

Java技術棧

相關文章
相關標籤/搜索