多線程編程之——終止(打斷)正在執行中的線程

来源:https://www.cnblogs.com/kakarotto-chen/archive/2023/11/25/17852977.html
-Advertisement-
Play Games

多線程編程之——終止(打斷)正在執行中的線程 ps:文字有點多,想看結果的,直接跳轉:《二》 一、基礎知識 1、我們基於spring開發,把線程都交給spring 把線程交給spring管理好不好? 將線程交給Spring管理是一個常見的做法,特別是在基於Spring的應用程式中。通過將線程納入Sp ...


多線程編程之——終止(打斷)正在執行中的線程

  • ps:文字有點多,想看結果的,直接跳轉:《二》

一、基礎知識

1、我們基於spring開發,把線程都交給spring

  • 把線程交給spring管理好不好?

將線程交給Spring管理是一個常見的做法,特別是在基於Spring的應用程式中。通過將線程納入Spring的管理範圍,你可以利用Spring的依賴註入和生命周期管理功能,更好地控制線程的生命周期和資源。這樣做也有助於避免手動管理線程帶來的潛在問題,比如線程泄露和資源管理不當。但是需要註意的是,要確保在使用Spring管理線程時,遵循最佳實踐,避免出現死鎖和性能問題。

還有個好處:可以把spring的bean註入到線程中,然後直接使用。(如註入service完成查詢數據)

如果把線程交給spring管理,註入線程的地方全是單例,如果要使用這個線程來完成兩個不同的任務,需要把單例變成多例模式,spring多例模式見:SpringBoot實現多例模式

  • 一個線程預設當做執行一個任務的最小單元

所以,當你需要執行一個任務的時候,應該把參數封裝到線程中,然後線上程中去執行你的任務(方法)

2、啟動Thread線程是調用start還是run?

在Java中,啟動一個Thread線程應該調用start方法而不是run方法。調用start方法會使Thread進入就緒狀態,並讓系統調度器來調用run方法。直接調用run方法只會在當前線程中執行run方法的代碼,而不會創建新的線程。因此,正確的做法是調用start方法來啟動一個新的線程。

3、打斷執行中的線程

  • 如果我new了一個Thread線程,它一直在執行中(無限阻塞),我後面調用了interrupt打斷線程,這個Thread線程會一直阻塞還是結束任務?

如果你在一個線程對象上調用interrupt方法,會設置線程的中斷狀態,並且如果線程正在阻塞中(比如在sleep、wait、join等方法中),它會拋出InterruptedException並清除中斷狀態。但如果線程正在執行普通的任務,它不會自動停止,而是需要在任務中檢查中斷狀態並做出相應的處理。因此,你需要在任務中定期檢查線程的中斷狀態,併在適當的時候結束任務。

  • 我調用interrupt的線程,如果一直處於睡眠中,jvm會自動回收嗎?

當你調用一個處於睡眠狀態的線程的interrupt方法時,它會拋出InterruptedException並清除中斷狀態,但並不會導致線程被自動回收。處於睡眠狀態的線程會繼續睡眠直到其睡眠時間結束或者被喚醒。如果你希望中斷一個處於睡眠狀態的線程並結束它的任務,你需要在任務中捕獲InterruptedException並做出相應的處理。

  • 捕獲InterruptedException併在捕獲到異常時結束任務或者進行清理工作。示例:
public class MyTask implements Runnable {
    public void run() {
        try {
            // 在這裡調用sleep方法使線程進入睡眠狀態(模仿無限阻塞)
            Thread.sleep(100000);
        } catch (InterruptedException e) {
            // 捕獲InterruptedException並做出相應的處理
            // 可以結束任務或者進行清理工作
            Thread.currentThread().interrupt(); // 重新設置中斷狀態
            return; // 結束任務
        }
        // 其他任務邏輯
    }
}

在上面的示例中,當線程處於睡眠狀態時,如果外部調用了interrupt方法,線程會拋出InterruptedException,然後在catch塊中重新設置中斷狀態並結束任務。
要想終止線程,必須線上程中調用:Thread.currentThread().interrupt();方法,而不是外部調用。

二、打斷線程

1、打斷new出來的線程

  • 也可以把線程交給spring管理
  • 直接new一個線程,然後在需要打斷的地方進行打斷,示例:
package com.cc.jschdemo.terminationThread;

/**
 * <p>基本打斷</p>
 *
 * @author cc
 * @since 2023/11/23
 */
public class BasicInterrupt {

    /**
     * 終止
     * @param args
     */
    public static void main(String[] args) {
        //t1:模仿執行任務的線程
        Thread t1 = new Thread(() -> {
            try {
                //步奏1
                Thread.sleep(1000);
                System.out.println("完成:步奏1");

                //步奏2(模仿阻塞)
                while (true) {
                    Thread.sleep(1000);
                    System.out.println("完成:步奏2");
                }
            }catch (InterruptedException e) {
                e.printStackTrace();
                //捕獲異常:InterruptedException,然後打斷
                //☆☆打斷,終止當前線程☆☆
                Thread.currentThread().interrupt();
            }
        });

        //t2:模仿其他操作,打斷線程
        Thread t2 = new Thread(() -> {
            try {
                Thread.sleep(3100);
                System.out.println("打斷線程!");
                //打斷t1線程
                t1.interrupt();
            }catch (Exception e) {
                e.printStackTrace();
            }
        });

        //啟動兩個線程
        t1.start();
        t2.start();
    }
}
  • 測試結果:步奏1執行1次,步奏2執行兩次,然後t1線程就會被t2線程終止
  • 線程被打斷會拋出錯誤:java.lang.InterruptedException: sleep interrupted

image

2、打斷註入spring的線程

  • 新建線程,註入spring中
  • ps:註意,註入spring的線程是單例的,實際使用過程中,我們都是一個線程一個任務,如果需要不同的對象,可以開啟spring的多例,見:SpringBoot實現多例模式
package com.cc.jschdemo.terminationThread;

import lombok.Data;
import lombok.EqualsAndHashCode;
import org.springframework.stereotype.Component;

/**
 * <p>註入spring的線程</p>
 *
 * @author --
 * @since 2023/11/24
 */
@EqualsAndHashCode(callSuper = true)
@Data
@Component
public class SpringThread extends Thread{

    //1可以註入Spring的Bean,然後直接使用
//    @Resource
//    private IBasicInterruptService basicInterruptService;

    //2可以直接傳入執行任務的參數(可多個)
    private String footwork;

    /**
     * 具體要執行的任務
     */
    @Override
    public void run() {
        try {
            //步奏1
            Thread.sleep(1000);
            System.out.printf("完成:%s 1 %n", footwork);

            //步奏2(模仿阻塞)
            while (true) {
                Thread.sleep(1000);
                System.out.printf("完成:%s 2 %n", footwork);
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
            //捕獲異常:InterruptedException,然後打斷
            //☆☆打斷,終止當前線程☆☆
            Thread.currentThread().interrupt();
        }
    }
}
  • 創建任務,模仿打斷
    //可以直接註入線程
    @Resource
    private SpringThread springThread;
    
    //模仿創建任務、打斷
    @Test
    public void test07()throws Exception{
        try {
            ///傳入執行任務,需要的參數
            springThread.setFootwork("步奏");
            //啟動線程
            springThread.start();

            Thread.sleep(3100);
            System.out.println("打斷線程!");
            //打斷t1線程
            springThread.interrupt();
        }catch (Exception e) {
            e.printStackTrace();
        }
    }
  • 結果:和《打斷new出來的線程》一模一樣

image

3、打斷CompletableFuture啟動的線程

  • CompletableFuture的cancel(true)方法無法打斷
  • 調用線程的Thread task1,的interrupt也無法打斷
  • CompletableFuture.runAsync(task1)啟動線程task1的底層邏輯是:CompletableFuture調用ForkJoinPool線程池啟動一個線程(A)去執行task1任務。
  • 所以我們要打斷的不是task1,而是要打斷A
package com.cc.jschdemo.terminationThread;

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.concurrent.CompletableFuture;

/**
 * <p></p>
 *
 * @author --
 * @since 2023/11/25
 */
@RestController
@RequestMapping("/completableFuture")
public class CompletableFutureThread {

    //這裡使用標記打斷的話,每個CompletableFuture都需要一個標記,可以(CompletableFuture和標記)一起緩存下來。
    boolean flagTask1 = false;
    boolean flagTask2 = false;

    //模仿任務1線程,也可以是:Runnable
    Thread task1 = new Thread(() -> {
        try {
            //模仿線程無限阻塞(while (true))
            while (true) {
                System.out.println("任務1執行...");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }

                //檢查終止狀態,如果終止就報錯。true已中斷;false未中斷
                // 這裡獲取到的是:CompletableFuture啟動線程池的線程池的線程。就是執行task1的線程
                boolean interrupted = Thread.currentThread().isInterrupted();
                // 所以interrupted不能用來終止task1

                //這裡使用標記來終止task1(可以根據業務,在需要終止的地方設置這個打斷。)
                if (flagTask1) {
                    //打斷:CompletableFuture啟動線程池的線程池的線程
                    Thread.currentThread().interrupt();
                }
            }
        }catch(Exception e){
            e.printStackTrace();
        }
    });

    //任務2
    Thread task2 = new Thread(() -> {
        try {
            //模仿線程阻塞
            while (true) {
                System.out.println("任務2執行...");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }

                //這裡使用標記來終止task1
                if (flagTask2) {
                    Thread.currentThread().interrupt();
                }
            }
        }catch(Exception e){
            e.printStackTrace();
        }
    });

    //緩存CompletableFuture
    CompletableFuture<Void> future1;
    CompletableFuture<Void> future2;

    //啟動任務
    @GetMapping
    public void test08()throws Exception{
        //實際使用中,可以使用迴圈添加CompletableFuture,然後記錄下future(緩存),在要中斷的邏輯中調用cancel
        //1開啟task任務。這裡實際是CompletableFuture調用ForkJoinPool線程池啟動一個線程去執行task1任務。
        //要想終止任務,要終止CompletableFuture調用ForkJoinPool線程池創建的線程,而非task1線程。
        future1 = CompletableFuture.runAsync(task1);
        future2 = CompletableFuture.runAsync(task2);
    }

    //終止任務
    @PostMapping
    public void stop() {
        //3取消任務
        // 參數true表示嘗試中斷任務執行
        // cancel方法會嘗試取消任務的執行,參數true表示嘗試中斷任務執行。
        // 成功取消返回true,否則返回false。
        // 需要註意的是,cancel方法並不會直接中斷正在執行的線程,而是會嘗試取消任務的執行,
        // 具體是否成功取決於任務的實現。

        //終止方式1:使用cancel終止:無法終止task1
        boolean cancel1 = future1.cancel(Boolean.TRUE);
        System.out.println("任務1終止:" + (cancel1 ? "成功1" : "失敗1"));
        //終止方式2:打斷線程task1:無法終止task1
        task1.interrupt();

        //睡5s,讓任務跑5s,看終止方式1、2是否能終止,測試結果為:不能終止
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }

        //終止方式3:直接標記打斷。CompletableFuture調用ForkJoinPool線程池生成的線程
        // 真正的終止task1
        flagTask1 = true;
    }
}



  • 測試

  • 終止方式1/2都執行了。但是任務沒有終止

image

  • 終止方式3執行後,才真正的終止了。

image


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • matplotlib是基於python生態開發的一個可視化繪圖庫,它的出現讓python在數據分析及機器學習方面占了重要的一部分,目前很多數據分析及機器學習相關方面的工程都有使用到這個庫,並且由於其簡單易用,安裝簡單等方面的優勢深得廣大開發者的喜愛。 ...
  • 15.1、常用組件 15.1.1、 DispatcherServlet DispatcherServlet 是前端控制器,由框架提供,不需要工程師開發; 作用:統一處理請求和響應,整個流程式控制制的中心,由它調用其它組件處理用戶的請求。 15.1.2、HandlerMapping HandlerMapp ...
  • 前言: 在Spring Boot中實現快遞鳥、順豐和快遞100的物流查詢功能通常需要與它們提供的API進行交互。當然使用他們的API 我們是需要申請和註冊,從而去拿到 key 來進行調用。所以為註冊的必須先進行註冊,以下是他們的官網地址,可以快捷到達。 快遞鳥官網:快遞鳥 - 快遞查詢介面_免費快遞 ...
  • SQLite,作為一款嵌入式關係型資料庫管理系統,一直以其輕量級、零配置以及跨平臺等特性而備受青睞。不同於傳統的資料庫系統,SQLite是一個庫,直接與應用程式一同編譯和鏈接,無需單獨的資料庫伺服器進程,實現了資料庫的零配置管理。這種設計理念使得SQLite成為許多嵌入式系統、移動應用和小型項目中的... ...
  • 操作系統 :CentOS 7.6_x64 FreeSWITCH版本 :1.10.9 python版本:3.9.12 libpcap版本:1.11.0b7 FreeSWITCH的ESL模塊用起來很方便,可以控制FreeSWITCH實現具體業務需求,但該模塊沒有提供ESL命令執行日誌,不便於排查問題,本 ...
  • 在本期文章中,我們深入探討了Python的對象繼承、組合以及多態這三個核心概念。從繼承的靈活性,如Python的多重繼承和super關鍵字的使用,到組合中的動態屬性添加,我們逐一解析了Python與Java在這些方面的相似之處和差異。通過具體的例子,我們展示了Python中多態的直觀表現,強調了它與... ...
  • API成批分配漏洞介紹API 特定:可利用性 2 利用通常需要瞭解業務邏輯、對象關係和 API 結構。 在 API 中利用批量分配更容易,因為按照設計,它們公開了應用程式的底層實現以及屬性名稱。安全弱點: 現代框架鼓勵開發人員使用自動將客戶端輸入綁定到代碼變數和內部對象的函數。 攻擊者可以使用這種方 ...
  • 十七、C++字元串(二) 1、字元串的應用 需求:設計一個程式,用戶輸入屬性id或者pass或者role可以把對應的內容顯示出來,給定字元串如下: string str{"id=user;pass=632105;role=郝英俊;"}; //設計一個程式,用戶輸入屬性id或者pass或者role可以 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...