国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

閉鎖和柵欄的區(qū)分以及適用場景

開發(fā) 開發(fā)工具
相信小伙伴對這個兩個詞或多或少都有些了解,他們是在并發(fā)編程中常用的線程通訊工具。兩者十分相似,但是又有不同,導致很多小伙伴也包括我在內(nèi)產(chǎn)生了很多困惑:他們兩個究竟有什么區(qū)別,以及適用于什么場景呢?

開篇

相信小伙伴對這個兩個詞或多或少都有些了解,他們是在并發(fā)編程中常用的線程通訊工具。兩者十分相似,但是又有不同,導致很多小伙伴也包括我在內(nèi)產(chǎn)生了很多困惑:他們兩個究竟有什么區(qū)別,以及適用于什么場景呢?

[[330337]]

下面聽我緩緩道來,不想看例子或者過程的小伙伴可以拉到最下面看總結呦

閉鎖

閉鎖(CountDownLatch)坊間俗稱計數(shù)器,官方(谷歌機翻,哈哈)解釋:

 

  1. /** 
  2.  * A synchronization aid that allows one or more threads to wait until 
  3.  * a set of operations being performed in other threads completes. 
  4.  */ 
  5.  
  6. 允許一個或多個線程等待,直到在其他線程中執(zhí)行的一組操作完成的同步輔助程序。 

大概意思就是說,可以有一個或者多個線程,等待其他線程都完成某個操作后,再繼續(xù)執(zhí)行。

什么意思呢?舉個栗子吧:

生活中應該經(jīng)常遇見一種情況,坐公交車是,尤其是始發(fā)站,司機師傅往往為了一次拉更多的乘客,會等到車上乘客的數(shù)量到達一定程度以后才會發(fā)車。測試代碼如下:

 

  1. public static void main(String[] args) { 
  2.         List<Passenger> list = new ArrayList<>(); 
  3.         Passenger p1 = new Passenger("看會書"); 
  4.         Passenger p2 = new Passenger("看會手機"); 
  5.         Passenger p3 = new Passenger("看會風景"); 
  6.         Passenger p4 = new Passenger("看會售票員"); 
  7.         list.add(p1); 
  8.         list.add(p2); 
  9.         list.add(p3); 
  10.         list.add(p4); 
  11.         ThreadPoolExecutor executor = new ThreadPoolExecutor(20, 200, 1000, TimeUnit.SECONDS, new LinkedBlockingQueue<>(3), new ThreadFactory() { 
  12.             private ThreadGroup group = (null == System.getSecurityManager() ? Thread.currentThread().getThreadGroup() : System.getSecurityManager().getThreadGroup()); 
  13.             private AtomicInteger num = new AtomicInteger(); 
  14.             @Override 
  15.             public Thread newThread(Runnable r) { 
  16.                 Thread thread = new Thread(group, r,"zoo" + num.getAndIncrement(),0); 
  17.                 thread.setDaemon(false); 
  18.                 return thread; 
  19.             } 
  20.         }, new ThreadPoolExecutor.CallerRunsPolicy()); 
  21.         //設定閉鎖釋放閾值 
  22.         CountDownLatch countDownLatch = new CountDownLatch(list.size()); 
  23.         log.error("司機師傅人夠一車再發(fā)車,等會人吧..."); 
  24.         for (Passenger p : list) { 
  25.             executor.execute(()->gotoZOO(p,countDownLatch)); 
  26.         } 
  27.         try { 
  28.             countDownLatch.await(); 
  29.             log.error("人夠了,起飛!"); 
  30.             executor.shutdown(); 
  31.         } catch (InterruptedException e) { 
  32.             e.printStackTrace(); 
  33.         } 
  34.  
  35.     } 
  36.  
  37.     private static void  gotoZOO(Passenger p,CountDownLatch countDownLatch){ 
  38.         log.error("{}的乘客上車啦",p.getDoWhat()); 
  39.         try { 
  40.             countDownLatch.countDown(); 
  41.             log.error("{}",p.doWhatOnBus()); 
  42.         } catch (Exception e) { 
  43.             e.printStackTrace(); 
  44.         } 
  45.     } 
  46.  
  47.     static class Passenger{ 
  48.         private String doWhat; 
  49.  
  50.         public Passenger(String doWhat) { 
  51.             this.doWhat = doWhat; 
  52.         } 
  53.  
  54.         public String getDoWhat() { 
  55.             return doWhat; 
  56.         } 
  57.  
  58.         public String doWhatOnBus() { 
  59.             return "車上好無聊啊,"+doWhat+"吧!"
  60.         } 
  61.     } 

執(zhí)行結果

 

  1. 23:46:34.698 [main] ERROR com.test - 司機師傅人夠一車再發(fā)車,等會人吧... 
  2. 23:46:34.757 [zoo1] ERROR com.test - 看會手機的乘客上車啦 
  3. 23:46:34.758 [zoo3] ERROR com.test - 看會售票員的乘客上車啦 
  4. 23:46:34.757 [zoo0] ERROR com.test - 看會書的乘客上車啦 
  5. 23:46:34.759 [zoo1] ERROR com.test - 車上好無聊啊,看會手機吧! 
  6. 23:46:34.759 [zoo3] ERROR com.test - 車上好無聊啊,看會售票員吧! 
  7. 23:46:34.757 [zoo2] ERROR com.test - 看會風景的乘客上車啦 
  8. 23:46:34.759 [zoo0] ERROR com.test - 車上好無聊啊,看會書吧! 
  9. 23:46:34.759 [zoo2] ERROR com.test - 車上好無聊啊,看會風景吧! 
  10. 23:46:34.759 [main] ERROR com.test - 人夠了,起飛! 

司機師傅(主線程)要等上了4個乘客以后才發(fā)車(等待4個子線程完成完成某件事以后調(diào)用countDown方法),而乘客上車(調(diào)用countDown)以后該做自己的事還做自己的事情,不會因為上了車就傻呆呆的什么都不干了(不會因為調(diào)用了countDown而阻塞自身)。等司機師傅看人夠了(到達設定閾值),就發(fā)車了。

閉鎖總結:

  • 主線程調(diào)用await后會阻塞等待其他子線程調(diào)用countDown方法將設定閾值減至0,然后在繼續(xù)執(zhí)行。
  • 而子線程不會因為調(diào)用了countDown方法而阻塞

柵欄

柵欄(CyclicBarrier)官方解釋:

 

  1. /** 
  2.  * A synchronization aid that allows a set of threads to all wait for 
  3.  * each other to reach a common barrier point.  CyclicBarriers are 
  4.  * useful in programs involving a fixed sized party of threads that 
  5.  * must occasionally wait for each other. The barrier is called 
  6.  * <em>cyclic</em> because it can be re-used after the waiting threads 
  7.  * are released. 
  8.  */ 
  9. 同步幫助,允許一組線程互相等待,以達到共同的障礙點。 CyclicBarriers在涉及固定大小的線程方的程序中很有用,這些線程有時必須互相等待。該屏障稱為<em> cyclic </ em>,因為它可以在釋放等待線程后重新使用。 

從類注釋上我們可以大致了解到,他是運用在一組,也即是多個線程中的,當所有線程到達某個狀態(tài)前一直阻塞,直到所有線程都達到后再繼續(xù)執(zhí)行。而且是可以重復使用的。

上面的描述還是太晦澀了,還是舉個栗子:

我們小時候學校都組織過春游,規(guī)定好地點,等人到齊了就一起進去玩。寫了個簡單的例子,看這種場景柵欄是怎么工作的

 

  1. public static void main(String[] args) { 
  2.         List<Boy> list = new ArrayList<>(); 
  3.         Boy boy1 = new Boy("看老虎"); 
  4.         Boy boy2 = new Boy("看猩猩"); 
  5.         Boy boy3 = new Boy("看獅子"); 
  6.         Boy boy4 = new Boy("看售票員"); 
  7.         list.add(boy1); 
  8.         list.add(boy2); 
  9.         list.add(boy3); 
  10.         list.add(boy4); 
  11.         ThreadPoolExecutor executor = new ThreadPoolExecutor(20, 200, 1000, TimeUnit.SECONDS, new LinkedBlockingQueue<>(3), new ThreadFactory() { 
  12.             private ThreadGroup group = (null == System.getSecurityManager() ? Thread.currentThread().getThreadGroup() : System.getSecurityManager().getThreadGroup()); 
  13.             private AtomicInteger num = new AtomicInteger(); 
  14.             @Override 
  15.             public Thread newThread(Runnable r) { 
  16.                 Thread thread = new Thread(group, r,"zoo" + num.getAndIncrement(),0); 
  17.                 thread.setDaemon(false); 
  18.                 return thread; 
  19.             } 
  20.         }, new ThreadPoolExecutor.CallerRunsPolicy()); 
  21.         //初始化柵欄,設置障礙點閾值 
  22.         CyclicBarrier cyclicBarrier = new CyclicBarrier(list.size()); 
  23.         for (Boy boy : list) { 
  24.             executor.execute(()->gotoZOO(boy,cyclicBarrier)); 
  25.         } 
  26.     } 
  27.  
  28.     private static void  gotoZOO(Boy boy,CyclicBarrier cyclicBarrier){ 
  29.         log.error("人還沒到齊呢,等一下吧,{}的小男孩開始等待",boy.getWhere()); 
  30.         try { 
  31.             cyclicBarrier.await(); 
  32.             log.error("{}",boy.goWhere()); 
  33.         } catch (Exception e) { 
  34.             e.printStackTrace(); 
  35.         } 
  36.     } 
  37.  
  38.     static class Boy{ 
  39.         private String where
  40.         public Boy(String where) { 
  41.             this.where = where
  42.         } 
  43.  
  44.         public String getWhere() { 
  45.             return where
  46.         } 
  47.  
  48.         public String goWhere() { 
  49.             return "人到齊了,我要去"+where+"啦!"
  50.         } 
  51.     } 

執(zhí)行結果:

 

  1. 22:05:59.476 [zoo2] ERROR com.test - 人還沒到齊呢,等一下吧,看獅子的小男孩開始等待 
  2. 22:05:59.477 [zoo1] ERROR com.test - 人還沒到齊呢,等一下吧,看猩猩的小男孩開始等待 
  3. 22:05:59.477 [zoo0] ERROR com.test - 人還沒到齊呢,等一下吧,看老虎的小男孩開始等待 
  4. 22:05:59.476 [zoo3] ERROR com.test - 人還沒到齊呢,等一下吧,看售票員的小男孩開始等待 
  5. 22:05:59.484 [zoo0] ERROR com.test - 人到齊了,我要去看老虎啦! 
  6. 22:05:59.484 [zoo2] ERROR com.test - 人到齊了,我要去看獅子啦! 
  7. 22:05:59.484 [zoo3] ERROR com.test - 人到齊了,我要去看售票員啦! 
  8. 22:05:59.484 [zoo1] ERROR com.test - 人到齊了,我要去看猩猩啦! 

我們可以發(fā)現(xiàn)前三個小男孩在到達以后都沒有進到動物園里,而是直到第四個小男孩來到以后,四個小男孩才進入動物園,在此之前每來一個小朋友就多一個小朋友等待(每個線程調(diào)用await方法),直到等待所有人到齊(線程阻塞等待達到柵欄障礙點4),各個小男孩再去繼續(xù)進入動物園看動物(各線程繼續(xù)執(zhí)行自己的任務)。就像是動物園大門的柵欄,買的是團體票,每次必須人到齊才放開讓小朋友進去一樣。

柵欄總結

各子線程相互等待,直到達到柵欄初始化時的閾值,則繼續(xù)執(zhí)行

區(qū)分以及個人理解

閉鎖:有點類似于一個統(tǒng)計功能(可能這也是為什么他俗稱計數(shù)器),主線程調(diào)用await方法阻塞等待統(tǒng)計結果,而子線程只負責在達到統(tǒng)計要求時調(diào)用countDown方法告訴主線程我好了,而不會阻塞本身;有一個負責接收結果(主線程)和一個或多個發(fā)送數(shù)量的(子線程);

柵欄:首先在線程調(diào)用await方法時會阻塞當前線程,其次個人理解他沒有類似像閉鎖那樣的主子的關系,他是各個線程相互等待,都到達某個點的時候,則繼續(xù)執(zhí)行。

適用場景

其實從上面的區(qū)分就能看出一些:如果是需要將多線程執(zhí)行完成與否的接口匯總到某一個線程中,然后再繼續(xù)執(zhí)行的情況,比如每條線程計算一個指標,都計算完成以后再計算所有指標的總和或者其他的,就可以使用閉鎖;

而如果只是各個線程需要等各個線程都完成了,再繼續(xù)自己的事,可以使用柵欄,比如ABC三個線程分別去獲取123三個指標,然后再A要取這三個數(shù)的平均數(shù),B要取總和,C要取方差,那就需要等ABC都先取完了123這三個指標,才能計算,這時候就可以用到柵欄了。

總結

這兩種都是非常好的線程通訊工具,不過細節(jié)還是有所差異。

總得來說就是:

  • 閉鎖是為了在某一條線程等待獲取到其他線程的執(zhí)行結果;
  • 而柵欄則是線程間的相互等待,然后再同時開始做各自的事情

最后

文中的代碼只是為了比較好的說明兩種工具的差異,寫的不好還請小伙伴們多多包涵,如果發(fā)現(xiàn)有哪點寫的不對的也歡迎大家伙們留言,我們共同進步!最后如果小伙伴覺得文章不錯,不妨動動小手點個贊再走,不要下次一定呦~

責任編輯:未麗燕 來源: segmentfault.com
相關推薦

2023-11-29 07:43:30

2021-12-03 18:03:06

算法場景Rsa

2015-01-16 11:30:07

Openstack分布式存儲

2011-05-26 15:10:15

靜態(tài)變量

2021-03-04 09:00:00

架構Lambda工具

2010-03-02 16:50:34

WCF返回值

2020-04-07 14:20:10

RabbitMMySQL數(shù)據(jù)庫

2019-06-17 16:17:03

存儲MySQL主流

2024-10-11 11:50:05

Redis適用場景

2018-08-15 09:48:27

數(shù)據(jù)庫Redis應用場景

2023-06-06 08:18:24

Kafka架構應用場景

2018-01-25 19:09:40

JavaThreadLocal線程

2021-08-06 10:43:56

Kubernetes容器

2009-06-05 10:43:29

struts2 checheckbox實例

2022-10-28 07:15:26

策略模式使用場景UML

2021-08-29 22:05:04

對象自動回收

2025-10-09 09:28:50

2021-08-16 13:54:23

大數(shù)據(jù)深信服

2020-10-29 07:16:26

布隆過濾器場景

2015-08-03 13:36:40

Docker技術優(yōu)勢應用場景
點贊
收藏

51CTO技術棧公眾號

久久精品国产亚洲夜色av网站| 欧美成人app| 国产精品99导航| 91精品国产免费| 51精品国产黑色丝袜高跟鞋| 992tv快乐视频| 91在线视频| 欧美1区2区3区4区| 国产成人综合在线观看| 色婷婷av一区二区三区大白胸| 欧美日韩电影在线观看| 小说区视频区图片区| 国产激情在线| 欧美日韩亚洲一区二区三区在线| 亚洲国产日日夜夜| 欧美一级片久久久久久久| 伊人影院在线视频| 亚洲女人av| 岛国av午夜精品| 国产精品人成电影| metart日本精品嫩模| 欧美一级大片在线免费观看| 欧美精品电影| 亚洲精选一区| 色av一区二区| 日本久久精品一区二区| 国产中文欧美日韩在线| 丁香另类激情小说| 日韩在线观看精品| 欧美日韩在线不卡视频| 9999在线精品视频| 久久久综合网站| 欧美成在线观看| 国产成人精品视频ⅴa片软件竹菊| 宅男噜噜噜66国产精品免费| 欧美女孩性生活视频| 日韩精品一区二区三区外面 | 国产成人亚洲精品狼色在线| 成人动漫在线视频| 成人av免费| 亚洲午夜久久久久久久久电影网| 成人xxxx视频| 91在线看黄| 性久久久久久久| 精品视频一区二区| 国内小视频在线看| 久久福利视频一区二区| 色哟哟网站入口亚洲精品| 熟妇人妻无乱码中文字幕真矢织江 | 国内精品久久久久影院 日本资源 国内精品久久久久伊人av | 国产成人精品亚洲日本在线桃色| 91精品国产91综合久久蜜臀| 国产一级免费在线观看| 久久国产精品色婷婷| 欧美精品v日韩精品v国产精品| 免费观看欧美大片| 亚洲精品国产视频| 国产经品一区二区| 久久精品嫩草影院| 亚洲一区二区三区四区五区中文| 成人黄网18免费观看的网站| 你懂的国产精品| 亚洲精品一区中文| 成人羞羞国产免费网站| 婷婷丁香综合| 精品亚洲一区二区三区在线观看| 丁香花在线电影小说观看| 精品国产一二三区| 波多结衣在线观看| 影音先锋久久| 日韩性生活视频| 性欧美video另类hd尤物| 欧美日韩免费网站| 男人的天堂在线| 国产99久久久国产精品潘金 | 国产裸体歌舞团一区二区| 97超级碰碰碰久久久| 在线国产91| 精品久久免费看| 日韩av电影免费| 久久成人综合网| 青青青国产在线观看| 韩日欧美一区| 国产+人+亚洲| 性xxxx欧美老肥妇牲乱| 亚洲性夜色噜噜噜7777| 欧美一区二区视频| 久久久久久一二三区| 成人影院在线观看视频| 精品国产福利视频| 男女视频一区二区三区| 亚洲激情六月丁香| 伊人精品影院| 色综合久久天天综线观看| 欧美一卡二卡| 亚洲成av人片在线| 在线观看黄av| 亚洲一区二区三区四区五区黄| 国产爆初菊在线观看免费视频网站| 成人黄色a**站在线观看| 亚洲japanese制服美女| 免费看日产一区二区三区| 国产精品久久久久久一区二区| 久久中文在线| 亚洲在线免费视频| 国内成+人亚洲+欧美+综合在线 | 麻豆久久一区二区| 91系列在线播放| 清纯唯美亚洲经典中文字幕| 国产视频一区二区不卡| 日韩电影一区| 97久久久久久| 国产精品美女久久久久| 亚洲欧美福利视频| 里番在线播放| 欧美有码在线观看| 久久黄色级2电影| 日本一卡二卡四卡精品| 亚洲观看高清完整版在线观看| 亚洲自拍电影| 亚洲欧美另类久久久精品| 99精品国产一区二区| 99国产精品久久久久久久成人热| 国产亚洲欧洲在线| 国产精品久久久久久久免费观看 | 欧美黑人性视频| 精品亚洲国产成人av制服丝袜| 手机av在线| 久久成人精品电影| 亚洲欧美春色| 天堂在线精品| 影音先锋男人的网站| 亚洲成人黄色影院| 亚洲国产精品一区二区尤物区| 精品一区二区三区视频| 欧美激情五月| 国产色99精品9i| 国产高清亚洲| 麻豆成全视频免费观看在线看| 日韩欧美一级二级| 亚洲1234区| 亚洲精品555| 天堂va欧美ⅴa亚洲va一国产| 免费的一级黄色片| 中文字幕中文字幕一区二区| 亚洲十八**毛片| 中文字幕 在线观看| 欧美人xxxxx| 国产欧美久久久久| 精品国产乱码久久久久久浪潮 | 91精品国产综合久久蜜臀| 欧美激情福利| 国产精品vvv| 久久男人av资源网站| 天天操天天干天天综合网| 中文字幕一区二区三区不卡在线| 国产日韩欧美a| 凹凸av导航大全精品| 最近中文字幕一区二区| 不卡影院一区二区| 国产剧情av在线| 国产精品一区二区久久久久| 洋洋av久久久久久久一区| 毛片在线播放a| 精品一区二区三区中文字幕 | 91大神精品| 欧美高清性xxxxhdvideosex| 一本久道久久综合狠狠爱亚洲精品| 91九色02白丝porn| 日韩一区在线视频| 中文综合在线观看| 亚洲国产精品麻豆| 午夜精品123| 色婷婷av一区二区三区软件 | 欧美日本韩国一区二区三区视频| 日夜干在线视频| 91国产美女视频| 国产女主播视频一区二区| 伪装者在线观看完整版免费| 精品成人佐山爱一区二区| 奇米在线7777在线精品| 国产偷人视频免费| 日韩亚洲国产中文字幕| 91一区在线观看| 国产在线观看免费| 国产美女99p| 精品国产一区二区精华| 国产乱码精品一区二区三区忘忧草 | 北条麻妃在线一区二区免费播放| 亚洲精品午夜在线观看| 色婷婷久久久久swag精品| 三上悠亚国产精品一区二区三区| 国产免费xxx| 麻豆国产va免费精品高清在线| 国产精品久久久久国产精品日日| 91精品国产视频| 美足av综合网| 亚洲 激情 在线| 国产精品三区www17con| 久久久不卡网国产精品一区|