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
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...