欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java多線程之Interrupt中斷線程詳解

 更新時間:2021年05月19日 16:20:49   作者:Ztrue  
Interrupt 的其作用是"中斷"線程, 但實際上線程仍會繼續(xù)運行, 這是一個非常容易混淆的概念. Interrupt 的真正作用是給線程對象設(shè)置一個中斷標(biāo)記, 并不會影響線程的正常運行,需要的朋友可以參考下

一、測試代碼

https://gitee.com/zture/spring-test/blob/master/multithreading/src/test/java/cn/diswares/blog/InterruptTests.java

二、測試

為了方便理解簡介中 interrupt 的概念, 寫個 DEMO 測試一下

 /**
 * 調(diào)用 interrupt 并不會影響線程正常運行
 */
@Test
public void testInvokeInterrupt() throws InterruptedException {
    Thread t1 = new Thread(() -> {
    for (int i = 0; ; i++) {
    log.info(i + "");
    }
    });
    t1.start();
    // 確保 t1.start() 成功執(zhí)行
    Thread.sleep(1);
    log.info("interrupt 前 t1 interrupt 狀態(tài) = {}", t1.isInterrupted());
    t1.interrupt();
    log.info("interrupt 后 t1 interrupt 狀態(tài) = {}", t1.isInterrupted());
    log.info("t1 是否存活 = {}", t1.isAlive());
}

三、執(zhí)行過程描述

  • 首先 main 線程中啟動 t1線程
  • t1 線程死循環(huán)輸出 i++
  • main 線程確保 t1.start() 執(zhí)行后
  • 打印 t1 線程的線程中斷狀態(tài)
  • 調(diào)用 t1.interrupt() 方法使線程中斷
  • 打印 t1 線程的線程中斷狀態(tài)

四、輸出日志

ignore logs ......
20:29:57.632 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2561
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2562
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2563
20:29:57.486 [main] INFO cn.diswares.blog.interrupt.InterruptTests - interrupt 前 t1 interrupt 狀態(tài) = false
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2564
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2565
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2566
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2567
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2568
20:29:57.633 [main] INFO cn.diswares.blog.interrupt.InterruptTests - interrupt 后 t1 interrupt 狀態(tài) = true
20:29:57.633 [main] INFO cn.diswares.blog.interrupt.InterruptTests - t1 是否存活 = true
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2569
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2570
20:29:57.633 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - 2571
ignore logs ......

現(xiàn)象描述

  • 調(diào)用 t1.interrupt() 執(zhí)行前線程的 interrupt 狀態(tài)為 false
  • 調(diào)用 t1.interrupt() 執(zhí)行后線程的 interrupt 狀態(tài)為 true
  • 線程并沒有被中斷, 可以成功死循環(huán)輸出循環(huán)次數(shù)

五、結(jié)論

Interrupt 的真正作用是給線程對象設(shè)置一個中斷標(biāo)記, 并不會影響線程的正常運行

六、主要方法釋義

new Thread().interrupt()

中斷此線程(此線程不一定是當(dāng)前線程,而是指調(diào)用該方法的Thread實例所代表的線程),但實際上只是給線程設(shè)置一個中斷標(biāo)志,線程仍會繼續(xù)運行。

Thread.interrupted()

注意: 這是個靜態(tài)方法
測試當(dāng)前線程是否被中斷(檢查中斷標(biāo)志), 返回一個當(dāng)前線程的 interrupt 狀態(tài), 并重置.
當(dāng)我們第二次調(diào)用時中斷狀態(tài)已經(jīng)被重置, 將返回一個false
為了方便理解. 寫一個 DEMO

七、DEMO

DEMO 非常簡單, 調(diào)用兩次 Thread.interrupted() 觀察 main 線程的 interrupt 標(biāo)記

/**
 * 二次調(diào)用 t1.interrupted()
 */
@Test
public void testDoubleInvokeInterrupted () throws InterruptedException {
    Thread.currentThread().interrupt();
    log.info("interrupted1 = {}", Thread.interrupted());
    log.info("interrupted2 = {}", Thread.interrupted());
}

輸出日志

21:06:33.397 [main] INFO cn.diswares.blog.interrupt.InterruptTests - interrupted1 = true
21:06:33.402 [main] INFO cn.diswares.blog.interrupt.InterruptTests - interrupted2 = false

八、拓展程序

由于是靜態(tài)方法. 我們來看一下另一個小程序.

  • 跟之前一樣將 t1 程序中斷
  • 調(diào)用 t1.interrupted()
  • 注意這里是個靜態(tài)方法
/**
 * 在主線程中調(diào)用 t1.interrupted()
 */
@Test
public void testMainInterrupted() throws InterruptedException {
    Thread t1 = new Thread(() -> {
        for (int i = 0; ; i++) {
            log.info("t1 is live");
        }
    });

    t1.start();
    Thread.sleep(1);
    t1.interrupt();
    Thread.sleep(1);
    log.info("{}", t1.interrupted());
}

拓展程序日志

ignore logs ......
21:11:20.504 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - t1 is live
21:11:20.504 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - t1 is live
21:11:20.490 [main] INFO cn.diswares.blog.interrupt.InterruptTests - false
21:11:20.504 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - t1 is live
21:11:20.504 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - t1 is live
21:11:20.504 [Thread-1] INFO cn.diswares.blog.interrupt.InterruptTests - t1 is live
ignore logs ......

拓展程序結(jié)論

  • Thread.interrupted() 方法是靜態(tài)方法
  • 它的實現(xiàn)為 Thread.currentThread(), 獲取的是當(dāng)前正在執(zhí)行的線程, JDK 原文注釋如下

Returns a reference to the currently executing thread object.

Returns: the currently executing thread.

  • 所以這里 t1.interrupted() 返回的其實是 main 線程的線程中斷標(biāo)記

new Thread().isInterrupted()

返回線程對象的中斷標(biāo)記, 不會改變中斷標(biāo)記

  • true: 中斷標(biāo)記存在
  • false: 未設(shè)置中斷標(biāo)記狀態(tài)

優(yōu)雅的結(jié)束一個線程

在 Java 中結(jié)束一個線程一般有下面三種手段:

  • (禁用) Thread.stop() 這個方法已經(jīng)被廢棄. 因為這種結(jié)束線程的方式過于暴力. 會將當(dāng)前線程暴力終結(jié). 同時線程持有的鎖也都會釋放, 并且用戶有任何額外的處理來控制, 會導(dǎo)致數(shù)據(jù)不一致
  • volatile: 外部申明 volatile 開關(guān)變量, 當(dāng)開關(guān)條件不滿足時結(jié)束
  • (推薦) interrupt: 最優(yōu)雅的方案

九、實戰(zhàn)

最初的 DEMO 是個死循環(huán), 那我們對其改造一下. 讓它能夠優(yōu)雅的結(jié)束

/**
 * 調(diào)用 interrupt 并不會影響線程正常運行
 */
@Test
public void testGracefulEndThread() throws InterruptedException {
    Thread t1 = new Thread(() -> {
        for (int i = 0; ; i++) {
            if (Thread.currentThread().isInterrupted()) {
                log.info("{} = true, i = {}", Thread.currentThread().getName(), i);
                break;
            } else {
                log.info("{} = false, i = {}", Thread.currentThread().getName(), i);
            }
        }
    });
    t1.start();
    // 確保 t1.start() 成功執(zhí)行
    TimeUnit.SECONDS.sleep(1);
    t1.interrupt();
    TimeUnit.SECONDS.sleep(1);
    log.info(t1.getState().toString());
}

到此這篇關(guān)于Java多線程之Interrupt中斷線程詳解的文章就介紹到這了,更多相關(guān)Java Interrupt中斷線程內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 自定義類加載器的父類為何是AppClassLoader說明

    自定義類加載器的父類為何是AppClassLoader說明

    這篇文章主要介紹了自定義類加載器的父類為何是AppClassLoader說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-11-11
  • 詳解Spring Cache使用Redisson分布式鎖解決緩存擊穿問題

    詳解Spring Cache使用Redisson分布式鎖解決緩存擊穿問題

    本文主要介紹了詳解Spring Cache使用Redisson分布式鎖解決緩存擊穿問題,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-04-04
  • Java中的Lambda表達(dá)式詳解

    Java中的Lambda表達(dá)式詳解

    這篇文章主要介紹了Java中的Lambda表達(dá)式詳解,Lambda 表達(dá)式是 JDK8 的一個新特性,可以取代大部分的匿名內(nèi)部類,寫出更優(yōu)雅的 Java 代碼,尤其在集合的遍歷和其他集合操作中,可以極大地優(yōu)化代碼結(jié)構(gòu),需要的朋友可以參考下
    2024-01-01
  • IntelliJ IDEA(2017)安裝和破解的方法

    IntelliJ IDEA(2017)安裝和破解的方法

    這篇文章主要介紹了IntelliJ IDEA(2017)安裝和破解的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-11-11
  • Java如何使用interrupt()終止線程

    Java如何使用interrupt()終止線程

    這篇文章主要介紹了Java如何使用interrupt()終止線程,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-03-03
  • Java中Spring使用Quartz任務(wù)調(diào)度定時器

    Java中Spring使用Quartz任務(wù)調(diào)度定時器

    本篇文章主要介紹了Java中Spring使用Quartz任務(wù)調(diào)度定時器,具有一定的參考價值,有興趣的可以了解一下。
    2017-02-02
  • Springboot集成百度地圖實現(xiàn)定位打卡的示例代碼

    Springboot集成百度地圖實現(xiàn)定位打卡的示例代碼

    本文主要介紹了Springboot集成百度地圖實現(xiàn)定位打卡的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-02-02
  • SpringCloud?Gateway實現(xiàn)API接口加解密

    SpringCloud?Gateway實現(xiàn)API接口加解密

    這篇文章主要為大家介紹了SpringCloud?Gateway如何實現(xiàn)API接口加解密的,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)有一定的幫助,需要的可以參考一下
    2022-06-06
  • 使用maven工具解決jar包沖突或重復(fù)加載的問題

    使用maven工具解決jar包沖突或重復(fù)加載的問題

    這篇文章主要介紹了使用maven工具解決jar包沖突或重復(fù)加載的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09
  • Java 如何優(yōu)雅的拷貝對象屬性

    Java 如何優(yōu)雅的拷貝對象屬性

    這篇文章主要介紹了Java 如何優(yōu)雅的拷貝對象屬性,幫助大家更好的理解和學(xué)習(xí)Java,感興趣的朋友可以了解下
    2020-11-11

最新評論