日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
一篇帶給你CountDownLatch實(shí)現(xiàn)原理

 前言

CountDownLatch是多線程中一個比較重要的概念,它可以使得一個或多個線程等待其他線程執(zhí)行完畢之后再執(zhí)行。它內(nèi)部有一個計(jì)數(shù)器和一個阻塞隊(duì)列,每當(dāng)一個線程調(diào)用countDown()方法后,計(jì)數(shù)器的值減少1。當(dāng)計(jì)數(shù)器的值不為0時,調(diào)用await()方法的線程將會被加入到阻塞隊(duì)列,一直阻塞到計(jì)數(shù)器的值為0。

目前創(chuàng)新互聯(lián)已為近1000家的企業(yè)提供了網(wǎng)站建設(shè)、域名、網(wǎng)站空間、成都網(wǎng)站托管、企業(yè)網(wǎng)站設(shè)計(jì)、南山網(wǎng)站維護(hù)等服務(wù),公司將堅(jiān)持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。

常用方法

 
 
 
 
  1. public class CountDownLatch { 
  2.  
  3.     //構(gòu)造一個值為count的計(jì)數(shù)器 
  4.     public CountDownLatch(int count); 
  5.  
  6.     //阻塞當(dāng)前線程直到計(jì)數(shù)器為0 
  7.     public void await() throws InterruptedException; 
  8.  
  9.     //在單位為unit的timeout時間之內(nèi)阻塞當(dāng)前線程 
  10.     public boolean await(long timeout, TimeUnit unit); 
  11.  
  12.     //將計(jì)數(shù)器的值減1,當(dāng)計(jì)數(shù)器的值為0時,阻塞隊(duì)列內(nèi)的線程才可以運(yùn)行 
  13.     public void countDown();       
  14.  

下面給一個簡單的示例:

 
 
 
 
  1. package com.yang.testCountDownLatch; 
  2.  
  3. import java.util.concurrent.CountDownLatch; 
  4.  
  5. public class Main { 
  6.     private static final int NUM = 3; 
  7.  
  8.     public static void main(String[] args) throws InterruptedException { 
  9.         CountDownLatch latch = new CountDownLatch(NUM); 
  10.         for (int i = 0; i < NUM; i++) { 
  11.             new Thread(() -> { 
  12.                 try { 
  13.                     Thread.sleep(2000); 
  14.                     System.out.println(Thread.currentThread().getName() + "運(yùn)行完畢"); 
  15.                 } catch (InterruptedException e) { 
  16.                     e.printStackTrace(); 
  17.                 } finally { 
  18.                     latch.countDown(); 
  19.                 } 
  20.             }).start(); 
  21.         } 
  22.         latch.await(); 
  23.         System.out.println("主線程運(yùn)行完畢"); 
  24.     } 

輸出如下:

看得出來,主線程會等到3個子線程執(zhí)行完畢才會執(zhí)行。

原理解析

類圖

 

可以看得出來,CountDownLatch里面有一個繼承AQS的內(nèi)部類Sync,其實(shí)是AQS來支持CountDownLatch的各項(xiàng)操作的。

CountDownLatch(int count)

new CountDownLatch(int count)用來創(chuàng)建一個AQS同步隊(duì)列,并將計(jì)數(shù)器的值賦給了AQS的state。

 
 
 
 
  1. public CountDownLatch(int count) { 
  2.     if (count < 0) throw new IllegalArgumentException("count < 0"); 
  3.     this.sync = new Sync(count); 
  4.  
  5. private static final class Sync extends AbstractQueuedSynchronizer {      
  6.     Sync(int count) { 
  7.         setState(count); 
  8.     } 
  9.  

countDown()

countDown()方法會對計(jì)數(shù)器進(jìn)行減1的操作,當(dāng)計(jì)數(shù)器值為0時,將會喚醒在阻塞隊(duì)列中等待的所有線程。其內(nèi)部調(diào)用了Sync的releaseShared(1)方法

 
 
 
 
  1. public void countDown() { 
  2.      sync.releaseShared(1); 
  3.  } 
  4.  
  5.  public final boolean releaseShared(int arg) { 
  6.      if (tryReleaseShared(arg)) { 
  7.          //此時計(jì)數(shù)器的值為0,喚醒所有被阻塞的線程 
  8.          doReleaseShared(); 
  9.          return true; 
  10.      } 
  11.      return false; 
  12.  } 

tryReleaseShared(arg)內(nèi)部使用了自旋+CAS操將計(jì)數(shù)器的值減1,當(dāng)減為0時,方法返回true,將會調(diào)用doReleaseShared()方法。對CAS機(jī)制不了解的同學(xué),可以先參考我的另外一篇文章淺探CAS實(shí)現(xiàn)原理

 
 
 
 
  1. protected boolean tryReleaseShared(int releases) { 
  2.       //自旋 
  3.       for (;;) { 
  4.           int c = getState(); 
  5.           if (c == 0) 
  6.               //此時計(jì)數(shù)器的值已經(jīng)為0了,其他線程早就執(zhí)行完畢了,當(dāng)前線程也已經(jīng)再執(zhí)行了,不需要再次喚醒了 
  7.               return false; 
  8.           int nextc = c-1; 
  9.           //使用CAS機(jī)制,將state的值變?yōu)閟tate-1 
  10.           if (compareAndSetState(c, nextc)) 
  11.               return nextc == 0; 
  12.       } 
  13.   } 

doReleaseShared()是AQS中的方法,該方法會喚醒隊(duì)列中所有被阻塞的線程。

 
 
 
 
  1. private void doReleaseShared() { 
  2.      for (;;) { 
  3.          Node h = head; 
  4.          if (h != null && h != tail) { 
  5.              int ws = h.waitStatus; 
  6.              if (ws == Node.SIGNAL) { 
  7.                  if (!compareAndSetWaitStatus(h, Node.SIGNAL, 0)) 
  8.                      continue;            // loop to recheck cases 
  9.                  unparkSuccessor(h); 
  10.              } 
  11.              else if (ws == 0 && 
  12.                       !compareAndSetWaitStatus(h, 0, Node.PROPAGATE)) 
  13.                  continue;                // loop on failed CAS 
  14.          } 
  15.          if (h == head)                   // loop if head changed 
  16.              break; 
  17.      } 
  18.  } 

這段方法比較難理解,會另外篇幅介紹。這里只要認(rèn)為該段方法會喚醒所有因調(diào)用await()方法而阻塞的線程。

await()

當(dāng)計(jì)數(shù)器的值不為0時,該方法會將當(dāng)前線程加入到阻塞隊(duì)列中,并把當(dāng)前線程掛起。

 
 
 
 
  1. public void await() throws InterruptedException { 
  2.     sync.acquireSharedInterruptibly(1); 

同樣是委托內(nèi)部類Sync,調(diào)用其

acquireSharedInterruptibly()方法

 
 
 
 
  1. public final void acquireSharedInterruptibly(int arg) 
  2.           throws InterruptedException { 
  3.       if (Thread.interrupted()) 
  4.           throw new InterruptedException(); 
  5.       if (tryAcquireShared(arg) < 0) 
  6.           doAcquireSharedInterruptibly(arg); 
  7.   } 

接著看Sync內(nèi)的tryAcquireShared()方法,如果當(dāng)前計(jì)數(shù)器的值為0,則返回1,最終將導(dǎo)致await()不會將線程阻塞。如果當(dāng)前計(jì)數(shù)器的值不為0,則返回-1。

 
 
 
 
  1. protected int tryAcquireShared(int acquires) { 
  2.         return (getState() == 0) ? 1 : -1; 
  3.     } 

tryAcquireShared方法返回一個負(fù)值時,將會調(diào)用AQS中的

doAcquireSharedInterruptibly()方法,將調(diào)用await()方法的線程加入到阻塞隊(duì)列中,并將此線程掛起。

 
 
 
 
  1. private void doAcquireSharedInterruptibly(int arg) 
  2.       throws InterruptedException { 
  3.       //將當(dāng)前線程構(gòu)造成一個共享模式的節(jié)點(diǎn),并加入到阻塞隊(duì)列中 
  4.       final Node node = addWaiter(Node.SHARED); 
  5.       boolean failed = true; 
  6.       try { 
  7.           for (;;) { 
  8.               final Node p = node.predecessor(); 
  9.               if (p == head) {         
  10.                   int r = tryAcquireShared(arg); 
  11.                   if (r >= 0) { 
  12.                       setHeadAndPropagate(node, r); 
  13.                       p.next = null; // help GC 
  14.                       failed = false; 
  15.                       return; 
  16.                   } 
  17.               } 
  18.               if (shouldParkAfterFailedAcquire(p, node) && 
  19.                   parkAndCheckInterrupt()) 
  20.                   throw new InterruptedException(); 
  21.           } 
  22.       } finally { 
  23.           if (failed) 
  24.               cancelAcquire(node); 
  25.       } 
  26.   } 

同樣,以上的代碼位于AQS中,在沒有了解AQS結(jié)構(gòu)的情況下去理解上述代碼,有些困難,關(guān)于AQS源碼,會另開篇幅介紹。

使用場景

CountDownLatch的使用場景很廣泛,一般用于分頭做某些事,再匯總的情景。例如:

數(shù)據(jù)報(bào)表:當(dāng)前的微服務(wù)架構(gòu)十分流行,大多數(shù)項(xiàng)目都會被拆成若干的子服務(wù),那么報(bào)表服務(wù)在進(jìn)行統(tǒng)計(jì)時,需要向各個服務(wù)抽取數(shù)據(jù)。此時可以創(chuàng)建與服務(wù)數(shù)相同的線程數(shù),交由線程池處理,每個線程去對應(yīng)服務(wù)中抽取數(shù)據(jù),注意需要在finally語句塊中進(jìn)行countDown()操作。主線程調(diào)用await()阻塞,直到所有數(shù)據(jù)抽取成功,最后主線程再進(jìn)行對數(shù)據(jù)的過濾組裝等,形成直觀的報(bào)表。

風(fēng)險評估:客戶端的一個同步請求查詢用戶的風(fēng)險等級,服務(wù)端收到請求后會請求多個子系統(tǒng)獲取數(shù)據(jù),然后使用風(fēng)險評估規(guī)則模型進(jìn)行風(fēng)險評估。如果使用單線程去完成這些操作,這個同步請求超時的可能性會很大,因?yàn)榉?wù)端請求多個子系統(tǒng)是依次排隊(duì)的,請求子系統(tǒng)獲取數(shù)據(jù)的時間是線性累加的。此時可以使用CountDownLatch,讓多個線程并發(fā)請求多個子系統(tǒng),當(dāng)獲取到多個子系統(tǒng)數(shù)據(jù)之后,再進(jìn)行風(fēng)險評估,這樣請求子系統(tǒng)獲取數(shù)據(jù)的時間就等于最耗時的那個請求的時間,可以大大減少處理時間。


網(wǎng)站標(biāo)題:一篇帶給你CountDownLatch實(shí)現(xiàn)原理
網(wǎng)站地址:http://www.5511xx.com/article/cogjsdo.html