Java線程通信

来源:https://www.cnblogs.com/taojietaoge/archive/2022/03/27/15997641.html
-Advertisement-
Play Games

Java線程通信 螣蛇乘霧,終為土灰。 多個線程協同工作完成某個任務時就會涉及到線程間通信問題。如何使各個線程之間同時執行,順序執行、交叉執行等。 一、線程同時執行 創建兩個線程a和b,兩個線程內調用同一個列印 1-3 三個數字的方法。 1 package tjt; 2 3 import java. ...


Java線程通信

 

      螣蛇乘霧,終為土灰。

 

多個線程協同工作完成某個任務時就會涉及到線程間通信問題。如何使各個線程之間同時執行,順序執行、交叉執行等。

一、線程同時執行

創建兩個線程a和b,兩個線程內調用同一個列印 1-3 三個數字的方法。

 1 package tjt;
 2 
 3 import java.time.LocalDate;
 4 
 5 public class Test {
 6 
 7     /**
 8      * 創建兩個線程a和b,兩個線程內調用同一個列印 1-3 三個數字的方法。
 9      */
10     private static void situationOne() {
11         Thread a = new Thread(new Runnable() {
12             @Override
13             public void run() {
14                 doSomething("a");
15             }
16         });
17         Thread b = new Thread(new Runnable() {
18             @Override
19             public void run() {
20                 doSomething("b");
21             }
22         });
23         a.start();
24         b.start();
25     }
26 
27     /**
28      * 依次列印 1, 2, 3 三個數字
29      *
30      * @param threadName
31      */
32     private static void doSomething(String threadName) {
33         int i = 0;
34         while (i++ < 3) {
35             try {
36                 Thread.sleep(200);
37             } catch (InterruptedException e) {
38                 e.printStackTrace();
39             }
40             System.out.println(LocalDate.now() + " Thread " + threadName + " is doing, printing: " + i);
41         }
42     }
43 
44     public static void main(String[] args) {
45         situationOne();
46     }
47 }
View Code

多次運行發現a和b是同時列印的,無執行順序可言。

二、線程順序執行

創建兩個線程a和b,要求b 在 a 全部列印完後再開始列印。使用 thread.join() 方法,在子線程調用了join()方法後面的代碼,只有等到子線程結束了才能執行,即必須a執行完畢後才輪到b。

 1 package tjt;
 2 
 3 import java.time.LocalDate;
 4 
 5 public class Test {
 6 
 7     /**
 8      * 創建兩個線程a和b,要求b 在 a 全部列印完後再開始列印,使用 thread.join() 方法。
 9      * 保證線程a執行完畢後才輪到b
10      */
11     private static void situationOne() {
12         Thread a = new Thread(new Runnable() {
13             @Override
14             public void run() {
15                 doSomething("a");
16             }
17         });
18         Thread b = new Thread(new Runnable() {
19             @Override
20             public void run() {
21                 try {
22                     System.out.println("線程 b 正在通過thread.join()等待線程 a 執行完畢後再潤");
23                     // thread.join() 在子線程調用了join()方法後面的代碼,只有等到子線程結束了才能執行,即必須a執行完畢後才輪到b
24                     a.join();
25                 } catch (InterruptedException e) {
26                     e.printStackTrace();
27                 }
28                 doSomething("b");
29             }
30         });
31         a.start();
32         b.start();
33     }
34 
35     /**
36      * 依次列印 1, 2, 3 三個數字
37      *
38      * @param threadName
39      */
40     private static void doSomething(String threadName) {
41         int i = 0;
42         while (i++ < 3) {
43             try {
44                 Thread.sleep(200);
45             } catch (InterruptedException e) {
46                 e.printStackTrace();
47             }
48             System.out.println(LocalDate.now() + " Thread " + threadName + " is doing, printing: " + i);
49         }
50     }
51 
52     public static void main(String[] args) {
53         situationOne();
54     }
55 }
View Code

無論運行多少次,都是線程a先執行完畢再到線程b。

三、線程順序交叉執行

創建兩個線程a和b,要求 a 在列印完 1 後,再讓 b 列印 1、2、 3,接著再回到 a 繼續列印 2、3。如此順序交叉執行僅靠 Thread.join() 是無法滿足需求的,需要更細粒度的鎖來控制執行順序,以及object.wait() 和 object.notify() 兩個方法來實現。

 1 package tjt;
 2 
 3 import java.time.LocalDate;
 4 
 5 public class TestAgain {
 6 
 7     private static void situationTwo() {
 8         // a 和 b 的共用對象鎖 lock
 9         Object lock = new Object();
10         Thread a = new Thread(new Runnable() {
11             @Override
12             public void run() {
13                 // 同步鎖 lock
14                 synchronized (lock) {
15                     // a 獲得鎖後執行
16                     doSomething("a", 1);
17                     try {
18                         // 調用 lock.wait() 方法,交出鎖的控制權,進入 wait 狀態,等待notify喚醒
19                         lock.wait();
20                     } catch (InterruptedException e) {
21                         e.printStackTrace();
22                     }
23                     doSomething("a", 2);
24                     doSomething("a", 3);
25                 }
26             }
27         });
28         Thread b = new Thread(new Runnable() {
29             @Override
30             public void run() {
31                 // 同步鎖 lock
32                 synchronized (lock) {
33                     // a 獲得鎖後執行
34                     doSomething("b", 1);
35                     doSomething("b", 2);
36                     doSomething("b", 3);
37                     // 調用 lock.notify() 方法,喚醒正在 wait 的線程 a
38                     lock.notify();
39                 }
40             }
41         });
42         a.start();
43         b.start();
44     }
45 
46     /**
47      * 列印
48      *
49      * @param threadName
50      * @param num
51      */
52     private static void doSomething(String threadName, int num) {
53         System.out.println(LocalDate.now() + " Thread " + threadName + " is doing, printing: " + num);
54     }
55 
56     public static void main(String[] args) {
57         situationTwo();
58     }
59 }
View Code

無論運行多少次,都是線程a先執行列印1,然後線程b執行列印1、2、3,最後線程a執行列印2、3。

四、CountDownLatch

CountDownLatch 計數器適用於一個線程去等待多個線程的情況。例如A B C 三個線程同時運行,各自獨立運行完後通知線程 D 執行,就可以利用 CountdownLatch 來實現這類通信方式。

對比之前的join方法,thread.join()可以讓一個線程等另一個線程運行完畢後再繼續執行,其可以在 D 線程里依次 join A B C,但這樣 A B C 必須依次執行,無法實現ABC三者能同步運行。 

 1 package tjt;
 2 
 3 import java.time.LocalDate;
 4 import java.util.concurrent.CountDownLatch;
 5 
 6 public class TestCountDownLatch {
 7 
 8     /**
 9      * countDownLatch 適用於一個線程去等待多個線程的情況
10      * 四個線程A、B、C、D,
11      * 其中 D 要等到 A B C 全執行完畢後才執行,且 A B C 是同步運行的,即ABC無順序執行
12      */
13     private static void situationThree() {
14         // 初始計數值設置為3,即總共四個線程A、B、C、D
15         CountDownLatch latch = new CountDownLatch(3);
16         new Thread(new Runnable() {
17             @Override
18             public void run() {
19                 System.out.println(LocalDate.now() + "線程 D 等待線程A B C 執行完畢後才可執行");
20                 try {
21                     // await() 檢查計數器值是否為 0,若不為 0 則保持等待狀態
22                     latch.await();
23                     // 其他線程 的 countDown() 方法把計數值變成 0 時,等待線程里的 countDownLatch.await() 立即退出,繼續執行下麵的代碼
24                     System.out.println(LocalDate.now() + "線程A B C 執行完畢,輪到線程D 執行了,當前latch:" + latch.getCount());
25                 } catch (InterruptedException e) {
26                     e.printStackTrace();
27                 }
28             }
29         }).start();
30 
31         // 迴圈執行線程 A B C
32         for (char threadName = 'A'; threadName <= 'C'; threadName++) {
33             String name = String.valueOf(threadName);
34             new Thread(new Runnable() {
35                 @Override
36                 public void run() {
37                     System.out.println("線程 " + name + " is running");
38                     // countDown(),將倒計數器減 1, 計數器被減至 0 時立即觸發D 的 await()
39                     try {
40                         Thread.sleep(200);
41                     } catch (InterruptedException e) {
42                         e.printStackTrace();
43                     }
44                     System.out.println("線程 " + name + " 執行完畢計數器");
45                     latch.countDown();
46                 }
47             }).start();
48         }
49     }
50 
51     public static void main(String[] args) {
52         situationThree();
53     }
54 
55 }
View Code

 五、CyclicBarrier 

實現線程間互相等待,可以利用 CyclicBarrier 柵欄。CountDownLatch 可以用來倒計數,但當計數完畢,只有一個線程的 await() 會得到響應,無法讓多個線程同時觸發。如要求線程 A B C 各自開始準備,直到三者都準備完畢再同時運行其就無法滿足需求,而用CyclicBarrier則完全OK。

 

 

 

 

 

 

螣蛇乘霧

終為土灰

 

 

 

 

 


 


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

-Advertisement-
Play Games
更多相關文章
  • 學習Flutter,首先需要搭建好Flutter的開發環境,下麵我將一步步帶領大家搭建開發環境並且成功運行flutter項目。 Flutter環境配置主要有這幾點: 系統配置要求 Java環境 Flutter SDK Android 開發環境 ***以下是基於在Window操作系統上安裝和配置Flu ...
  • 一、變數與屬性的區別; 1. 概念 什麼是變數? 變數:變數是一個存儲數據的容器,本質:變數是記憶體裡面的一個空間,通過變數名來獲取數據;聲明(var)一個變數就是在記憶體裡面獲取一個存儲空間。 什麼是屬性? 為元素提供各種附加信息的就是屬性,存在對象裡面表示對象有什麼樣的特征,常用名詞來表示,它總是以 ...
  • Array.find((item,indexArr,arr)=>{}) 掌握 找出第一個符合條件的數組成員。 它的參數是一個回調函數,對所有數組成員依次執行該回調函數。 直到找出第一個返回值為true的成員,然後返回該成員。 如果沒有符合條件的成員,則返回undefined。 -- 找出第一個大於1 ...
  • 客戶端通過虛擬ip(192.168.0.45)進行訪問haproxy,通過keepalived會將其發放到MASTER節點上的haproxy,若MASTER節點上的Haproxy宕機,則會發放到SLAVE節點上,通過Haproxy會訪問到RabbitMQ集群,通過集群訪問各個RabbitMQ ...
  • 不知大家可曾想過這樣一些問題:“在基於XML或基於Java的顯示配置中,假如需要配置的Bean很多,這時把所有Bean的配置信息全都填在一個配置文件里,難道不會導致配置文件非常臃腫,非常難以維護嗎?還有,假如因為某些原因不得不把一些Bean的配置信息填在XML配置文件里,把一些Bean的配置信息填在 ...
  • HashMap:是基於哈希表的Map介面實現。 * 哈希表的作用是用來保證鍵的唯一性的。 * * HashMap<String,String> * 鍵:String * 值:String */ package Day18; import java.util.HashMap; import java. ...
  • 博客推行版本更新,成果積累制度,已經寫過的博客還會再次更新,不斷地琢磨,高質量高數量都是要追求的,工匠精神是學習必不可少的精神。因此,大家有何建議歡迎在評論區踴躍發言,你們的支持是我最大的動力,你們敢投,我就敢肝 ...
  • 鍵值遍歷 package cn.itcast_01; import java.util.HashMap; import java.util.Map; import java.util.Set; /* * Map集合的遍歷。 * Map -- 夫妻對 * 思路: * A:把所有的丈夫給集中起來。 * ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...