摘要:考慮大量線程運行在一次計算的不同部分的情形。一旦所有的線程都到達了這個柵欄,柵欄就撤銷,線程可以繼續(xù)運行。那些已經在等待的線程立即中止的調用。如果在執(zhí)行屏障操作過程中發(fā)生異常,則該異常將傳播到當前線程中,并將置于損壞狀態(tài)。
【同步器
java.util.concurrent包包含幾個能幫助人們管理相互合作的線程集的類。這些機制具有為線程直間的共用集結點模式提供的‘預制功能’。如果有一個相互合作的線程滿足這些行為模式之一,那么應該直接使用提供的類庫而不是顯示的使用鎖與條件的集合。
【柵欄CyclicBarrier類實現了一個集結點(rendezvous)稱為柵欄(barrier)??紤]大量線程運行在一次計算的不同部分的情形。當所有部分都準備好時,需要把結果組合到一起。當一個線程完成了它那部分的任務后,我們讓他運行到柵欄處。一旦所有的線程都到達了這個柵欄,柵欄就撤銷,線程可以繼續(xù)運行。如果任何一個線程在柵欄上等待時離開,那么柵欄就被破壞掉(線程離開可能時等待超時)。這種情況下,其他所有線程await方法上拋出BrokenBarrierException異常。那些已經在等待的線程立即中止await的調用??梢蕴峁┮粋€可選的柵欄動作,當所有線程到達柵欄時,就會執(zhí)行這個動作。
Runnable barrierAction = ... CyclicBarrier barrire = new CyclicBarrier(nthreads,barrierAction);
柵欄被稱為時循環(huán)的,以為可以在所有等待的線程被釋放后重用(這里與倒計時門閂不同,倒計時門閂只能使用一次)。
【常用方法
public int await() throws InterruptedException,BrokenBarrierException:
在所有參與者都已經在此 barrier 上調用 await方法之前,將一直等待。如果當前線程不是將到達的最后一個線程,出于調度目的,將禁用它,且在發(fā)生以下情況之一前,該線程將一直處于休眠狀態(tài):
- 最后一個線程到達;或者 - 其他某個線程中斷當前線程;或者 - 其他某個線程中斷另一個等待線程;或者 - 其他某個線程在等待 barrier 時超時;或者 - 其他某個線程在此 barrier 上調用 reset()。
如果當前線程在進入此方法時已經設置了該線程的中斷狀態(tài);或者在等待時被中斷則InterruptedException,并且清除當前線程的已中斷狀態(tài)。
如果在線程處于等待狀態(tài)時 barrier 被 reset(),或者在調用 await 時 barrier 被損壞,抑或任意一個線程正處于等待狀態(tài),則拋出 BrokenBarrierException 異常。
如果任何線程在等待時被 中斷,則其他所有等待線程都將拋出 BrokenBarrierException 異常,并將 barrier 置于損壞狀態(tài)。
如果當前線程是最后一個將要到達的線程,并且構造方法中提供了一個非空的屏障操作,則在允許其他線程繼續(xù)運行之前,當前線程將運行該操作。
如果在執(zhí)行屏障操作過程中發(fā)生異常,則該異常將傳播到當前線程中,并將 barrier 置于損壞狀態(tài)。
返回:到達的當前線程的索引,其中,索引 getParties() - 1 指示將到達的第一個線程,零指示最后一個到達的線程.
拋出:
InterruptedException - 如果當前線程在等待時被中斷
BrokenBarrierException - 如果另一個 線程在當前線程等待時被中斷或超時,或者重置了 barrier,或者在調用 await 時 barrier 被損壞,抑或由于異常而導致屏障操作(如果存在)失敗。
【例子1.應用程序啟動前,所有子線程準備工作完成,不需要返回值
public class Application{ public static void main(String[] args) throws IOException, InterruptedException { //需要三個子線程準備 CyclicBarrier barrier = new CyclicBarrier(3); //如果初始化barrier的線程數,與實際子線程數不同,barrier將會一直等待 ExecutorService executor = Executors.newFixedThreadPool(3); executor.submit(new Thread(new InitThread(barrier, "子線程1"))); executor.submit(new Thread(new InitThread(barrier, "子線程2"))); executor.submit(new Thread(new InitThread(barrier, "子線程3"))); executor.shutdown(); } } class InitThread implements Runnable { // 一個同步輔助類,它允許一組線程互相等待,直到到達某個公共屏障點 (common barrier point) private CyclicBarrier barrier; private String name; public InitThread(CyclicBarrier barrier, String name) { this.barrier = barrier; this.name = name; } @Override public void run() { try { Thread.sleep(1000 * (new Random()).nextInt(8)); System.out.println(name + " 初始化完成..."); barrier.await(); } catch (InterruptedException e) { e.printStackTrace(); } catch (BrokenBarrierException e) { e.printStackTrace(); } System.out.println(name + " 開始運行!"); } }
2.應用程序啟動前,所有子線程準備工作完成,需要返回值匯總
public class ApplicationNeedReturn { public static void main(String[] args) throws IOException, InterruptedException, ExecutionException { //需要三個子線程準備 CyclicBarrier barrier = new CyclicBarrier(3); //如果初始化barrier的線程數,與實際子線程數不同,barrier將會一直等待 ExecutorService executor = Executors.newFixedThreadPool(3); FutureTasktime1 = new FutureTask (new InitThread2(barrier, "子線程1")); FutureTask time2 = new FutureTask (new InitThread2(barrier, "子線程2")); FutureTask time3 = new FutureTask (new InitThread2(barrier, "子線程3")); executor.submit(time1); executor.submit(time2); executor.submit(time3); long total = time1.get() + time2.get() + time3.get(); System.out.println("準備完成,耗時:" + total); executor.shutdown(); } } class InitThread2 implements Callable { private CyclicBarrier barrier; private String name; public InitThread2(CyclicBarrier barrier, String name) { this.barrier = barrier; this.name = name; } @Override public Long call() { long time = 1000 * (new Random()).nextInt(8); try { Thread.sleep(time); System.out.println(name + " 初始化完成... : " + time) ; barrier.await(); } catch (InterruptedException e) { e.printStackTrace(); } catch (BrokenBarrierException e) { e.printStackTrace(); } return time; } }
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/68089.html
摘要:整個包,按照功能可以大致劃分如下鎖框架原子類框架同步器框架集合框架執(zhí)行器框架本系列將按上述順序分析,分析所基于的源碼為。后,根據一系列常見的多線程設計模式,設計了并發(fā)包,其中包下提供了一系列基礎的鎖工具,用以對等進行補充增強。 showImg(https://segmentfault.com/img/remote/1460000016012623); 本文首發(fā)于一世流云專欄:https...
摘要:本文探討并發(fā)中的其它問題線程安全可見性活躍性等等。當閉鎖到達結束狀態(tài)時,門打開并允許所有線程通過。在從返回時被叫醒時,線程被放入鎖池,與其他線程競爭重新獲得鎖。 本文探討Java并發(fā)中的其它問題:線程安全、可見性、活躍性等等。 在行文之前,我想先推薦以下兩份資料,質量很高:極客學院-Java并發(fā)編程讀書筆記-《Java并發(fā)編程實戰(zhàn)》 線程安全 《Java并發(fā)編程實戰(zhàn)》中提到了太多的術語...
摘要:今天給大家總結一下,面試中出鏡率很高的幾個多線程面試題,希望對大家學習和面試都能有所幫助。指令重排在單線程環(huán)境下不會出先問題,但是在多線程環(huán)境下會導致一個線程獲得還沒有初始化的實例。使用可以禁止的指令重排,保證在多線程環(huán)境下也能正常運行。 下面最近發(fā)的一些并發(fā)編程的文章匯總,通過閱讀這些文章大家再看大廠面試中的并發(fā)編程問題就沒有那么頭疼了。今天給大家總結一下,面試中出鏡率很高的幾個多線...
摘要:分層支持分層一種樹形結構,通過構造函數可以指定當前待構造的對象的父結點。當一個的參與者數量變成時,如果有該有父結點,就會將它從父結點中溢移除。當首次將某個結點鏈接到樹中時,會同時向該結點的父結點注冊一個參與者。 showImg(https://segmentfault.com/img/remote/1460000016010947); 本文首發(fā)于一世流云專欄:https://segme...
摘要:的線程機制是搶占式。會讓出當多個線程并發(fā)的對主存中的數據進行操作時,有且只有一個會成功,其余均失敗。和對象只有在困難的多線程問題中才是必須的。 并發(fā)簡述 并發(fā)通常是用于提高運行在單處理器上的程序的性能。在單 CPU 機器上使用多任務的程序在任意時刻只在執(zhí)行一項工作。 并發(fā)編程使得一個程序可以被劃分為多個分離的、獨立的任務。一個線程就是在進程中的一個單一的順序控制流。java的線程機制是...
閱讀 2656·2023-04-26 00:07
閱讀 2439·2021-11-15 11:37
閱讀 650·2021-10-19 11:44
閱讀 2178·2021-09-22 15:56
閱讀 1735·2021-09-10 10:50
閱讀 1510·2021-08-18 10:21
閱讀 2578·2019-08-30 15:53
閱讀 1638·2019-08-30 11:11