java多線程-CyclicBarrier

来源:http://www.cnblogs.com/lcngu/archive/2016/02/24/5215332.html
-Advertisement-
Play Games

介紹 一個同步輔助類,它允許一組線程互相等待,直到到達某個公共屏障點 (common barrier point)。在涉及一組固定大小的線程的程式中,這些線程必須不時地互相等待,此時 CyclicBarrier 很有用。因為該 barrier 在釋放等待線程後可以重用,所以稱它為迴圈 的 barri


  • 介紹

  一個同步輔助類,它允許一組線程互相等待,直到到達某個公共屏障點 (common barrier point)。在涉及一組固定大小的線程的程式中,這些線程必須不時地互相等待,此時 CyclicBarrier 很有用。因為該 barrier 在釋放等待線程後可以重用,所以稱它為迴圈 的 barrier。 

  CyclicBarrier 支持一個可選的 Runnable 命令,在一組線程中的最後一個線程到達之後(但在釋放所有線程之前),該命令只在每個屏障點運行一次。若在繼續所有參與線程之前更新共用狀態,此屏障操作 很有用。 

  • 主要方法
  • 1 //設置parties、count及barrierCommand屬性。   
    2 CyclicBarrier(int):   
    3 //當await的數量到達了設定的數量後,首先執行該Runnable對象。   
    4 CyclicBarrier(int,Runnable):   
    5 //通知barrier已完成線程   
    6 await():  
  • 應用場景

  1:CyclicBarrier 表示大家彼此等待,大家集合好後才開始出發,分散活動後又在i指定地點集合碰面,這就好比整個公司的人員利用周末時間集體郊游一樣,先各自從家出發到公司集合後,再同時出發到公園游玩,在指定地點集合後再同時開始就餐。

  代碼:

 1 public class CyclicBarrierTest {
 2     public static void main(String[] args){
 3         ExecutorService pool = Executors.newCachedThreadPool();
 4         final CyclicBarrier cyclicBarrier = new CyclicBarrier(3);
 5         
 6         for (int i = 0; i < 3; i++) {
 7             Runnable runnable = new Runnable() {
 8                 @Override
 9                 public void run(){
10                     try {
11                         Thread.sleep(new Random().nextInt(5000));
12                     } catch (InterruptedException e) {
13                         // TODO Auto-generated catch block
14                         e.printStackTrace();
15                     }
16                     System.out.println(Thread.currentThread().getName()+"到達地點一,當前等待人數為"+(cyclicBarrier.getNumberWaiting()+1)+(cyclicBarrier.getNumberWaiting()+1==3?"繼續出發":"繼續等待"));
17                     try {
18                         cyclicBarrier.await();//障礙等待點
19                     } catch (InterruptedException e) {
20                         // TODO Auto-generated catch block
21                         e.printStackTrace();
22                     } catch (BrokenBarrierException e) {
23                         // TODO Auto-generated catch block
24                         e.printStackTrace();
25                     }
26                     try {
27                         Thread.sleep(new Random().nextInt(5000));
28                     } catch (InterruptedException e) {
29                         // TODO Auto-generated catch block
30                         e.printStackTrace();
31                     }
32                     System.out.println(Thread.currentThread().getName()+"到達地點二,當前等待人數為"+(cyclicBarrier.getNumberWaiting()+1)+(cyclicBarrier.getNumberWaiting()+1==3?"繼續出發":"繼續等待"));
33                     try {
34                         cyclicBarrier.await();//障礙等待點
35                     } catch (InterruptedException e) {
36                         // TODO Auto-generated catch block
37                         e.printStackTrace();
38                     } catch (BrokenBarrierException e) {
39                         // TODO Auto-generated catch block
40                         e.printStackTrace();
41                     }
42                     try {
43                         Thread.sleep(new Random().nextInt(5000));
44                     } catch (InterruptedException e) {
45                         // TODO Auto-generated catch block
46                         e.printStackTrace();
47                     }
48                     System.out.println(Thread.currentThread().getName()+"到達地點三,當前等待人數為"+(cyclicBarrier.getNumberWaiting()+1)+(cyclicBarrier.getNumberWaiting()+1==3?"人齊了出發":"繼續等待"));
49                     try {
50                         cyclicBarrier.await();//障礙等待點
51                     } catch (InterruptedException e) {
52                         // TODO Auto-generated catch block
53                         e.printStackTrace();
54                     } catch (BrokenBarrierException e) {
55                         // TODO Auto-generated catch block
56                         e.printStackTrace();
57                     }
58                 }
59             };
60             pool.execute(runnable);
61         }
62         pool.shutdown();
63     }
64 }

  執行結果:

1 pool-1-thread-3到達地點一,當前等待人數為1繼續等待
2 pool-1-thread-1到達地點一,當前等待人數為2繼續等待
3 pool-1-thread-2到達地點一,當前等待人數為3繼續出發
4 pool-1-thread-1到達地點二,當前等待人數為1繼續等待
5 pool-1-thread-3到達地點二,當前等待人數為2繼續等待
6 pool-1-thread-2到達地點二,當前等待人數為3繼續出發
7 pool-1-thread-3到達地點三,當前等待人數為1繼續等待
8 pool-1-thread-2到達地點三,當前等待人數為2繼續等待
9 pool-1-thread-1到達地點三,當前等待人數為3人齊了出發
View Code

  2:在某種需求中,比如一個大型的任務,常常需要分配好多子任務去執行,只有當所有子任務都執行完成時候,才能執行主任務,這時候,就可以選擇CyclicBarrier了。

  代碼:

 1 public class CyclicBarrierTest1 {
 2     public static void main(String[] args) {
 3         ExecutorService threadPool = Executors.newCachedThreadPool();
 4         CyclicBarrier barrier = new CyclicBarrier(5, new mainTask());
 5         for (int i = 0; i < 5; i++) {
 6             subTask subTask = new subTask(barrier);
 7             threadPool.execute(subTask);
 8         }
 9         threadPool.shutdown();
10     }
11 }
12 
13 class subTask implements Runnable{
14     private CyclicBarrier barrier;
15     
16     public subTask(CyclicBarrier barrier) {
17         super();
18         this.barrier = barrier;
19     }
20     @Override
21     public void run() {
22         System.out.println(Thread.currentThread().getName()+"正在執行");
23         try {
24             Thread.sleep(5000);
25         } catch (InterruptedException e) {
26             // TODO Auto-generated catch block
27             e.printStackTrace();
28         }
29         System.out.println(Thread.currentThread().getName()+"執行完畢,等待其他結果");
30         try {
31             barrier.await();
32         } catch (InterruptedException e) {
33             // TODO Auto-generated catch block
34             e.printStackTrace();
35         } catch (BrokenBarrierException e) {
36             // TODO Auto-generated catch block
37             e.printStackTrace();
38         }
39         
40     }
41 }
42 class mainTask implements Runnable{
43 
44     @Override
45     public void run() {
46         System.out.println("總任務執行完畢");
47     }
48     
49 }

  執行結果:

 1 pool-1-thread-2正在執行
 2 pool-1-thread-3正在執行
 3 pool-1-thread-1正在執行
 4 pool-1-thread-4正在執行
 5 pool-1-thread-5正在執行
 6 pool-1-thread-2執行完畢,等待其他結果
 7 pool-1-thread-5執行完畢,等待其他結果
 8 pool-1-thread-1執行完畢,等待其他結果
 9 pool-1-thread-4執行完畢,等待其他結果
10 pool-1-thread-3執行完畢,等待其他結果
11 總任務執行完畢
View Code

  另外,CyclicBarrier是可以重用的,它可以在使用完後繼續使用,這就是Cyclic(迴圈)的意思。

 


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

-Advertisement-
Play Games
更多相關文章
  • 先上題:下列運算符都可以被友元函數重載的是: A)=,+,-,\ B)[],+,(),new C)->,+,*,>> D)<<,>>,+,* 正確答案為D 我們知道,在運算符重載,友元函數運算符重載函數與成員運算符重載函數的區別是:友元函數沒有this指針,而成員函數有,因此,在兩個操作數的重載中友
  • login.jsp: 1 <%@ page language="java" import="java.util.*" contentType="text/html; charset=utf-8"%> 2 <% 3 String path = request.getContextPath(); 4 S
  • 在Java中,如果類的一個構造方法想要調用另一個構造方法,需要用this(參數列表)的形式來調用。 public class Student { String name; int age; public Student(int age) { this("Jack",age); } public St
  • Github連接 : 一、題目 題目描述連接: 二、準備過程 1. 把c++遠征計劃中的離港篇和封裝篇上看完了。( 學習計劃並沒有完成T_T ) 2. 發現題目中要求用到queue的知識,就去度娘了有關隊列的使用,大概知道該怎麼用吧。 3. 本來在電腦里下了c++ primer plus看一部分,發
  • 安裝JDK JDK的官網網址如下: http://www.oracle.com/technetwork/java/javase/downloads/index-jsp-138363.html 配置JDK環境變數參數設置 (1)JAVA_HOME: D:\Program Files (x86)\Jav
  • 眾所周知,json_encode通常會把json中的漢字轉義成unicode,但是這並不一定是我們想要的。有時候,我們需要獲得漢字形式的json字元串,比如需要獲得gbk編碼的json字元串(只要把漢字形式的字元串轉碼就可以得到了)。有什麼好辦法麽? php官方聽到了這個需求,並提供了一種可靠的解決
  • 簡介 一個同步輔助類,在完成一組正在其他線程中執行的操作之前,它允許一個或多個線程一直等待。用給定的計數 初始化 CountDownLatch。由於調用了 countDown() 方法,所以在當前計數到達零之前,await 方法會一直受阻塞。之後,會釋放所有等待的線程,await 的所有後續調用都將
  • IOC:即控制反轉,主要意思就是Spring容器來管理對象的初始化,而不需要程式員人工的使用new方式來創建對象,並且當A對象依賴於B對象時,在配置文件中可以指定,同樣不需要程式員在構造函數或是setter中進行對象註入。 AOP:面向切麵編程。其實就是一種新的不同於繼承的代碼重用技術。繼承是將共性
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...