欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談java并發(fā)之計數(shù)器CountDownLatch

 更新時間:2019年06月06日 11:28:53   作者:exposure  
CountDownLatch是通過一個計數(shù)器來實現(xiàn)的,當我們在new 一個CountDownLatch對象的時候需要帶入該計數(shù)器值,該值就表示了線程的數(shù)量。下面我們來深入了解一下吧

CountDownLatch簡介

CountDownLatch顧名思義,count + down + latch = 計數(shù) + 減 + 門閂(這么拆分也是便于記憶=_=) 可以理解這個東西就是個計數(shù)器,只能減不能加,同時它還有個門閂的作用,當計數(shù)器不為0時,門閂是鎖著的;當計數(shù)器減到0時,門閂就打開了。

如果你感到懵比的話,可以類比考生考試交卷,考生交一份試卷,計數(shù)器就減一。直到考生都交了試卷(計數(shù)器為0),監(jiān)考老師(一個或多個)才能離開考場。至于考生是否做完試卷,監(jiān)考老師并不關(guān)注。只要都交了試卷,他就可以做接下來的工作了。

CountDownLatch實現(xiàn)原理

下面從構(gòu)造方法開始,一步步解釋實現(xiàn)的原理:構(gòu)造方法下面是實現(xiàn)的源碼,非常簡短,主要是創(chuàng)建了一個Sync對象。

public CountDownLatch(int count) {
if (count < 0) throw new IllegalArgumentException("count < 0");
this.sync = new Sync(count);
}

Sync對象

private static final class Sync extends AbstractQueuedSynchronizer {
private static final long serialVersionUID = 4982264981922014374L;
Sync(int count) {
setState(count);
}
int getCount() {
return getState();
}
protected int tryAcquireShared(int acquires) {
return (getState() == 0) ? 1 : -1;
}
protected boolean tryReleaseShared(int releases) {
// Decrement count; signal when transition to zero
for (;;) {
int c = getState();
if (c == 0)
return false;
int nextc = c-1;
if (compareAndSetState(c, nextc))
return nextc == 0;
}
}
}

假設(shè)我們是這樣創(chuàng)建的:new CountDownLatch(5)。其實也就相當于new Sync(5),相當于setState(5)。setState其實就是共享鎖資源總數(shù),我們可以暫時理解為設(shè)置一個計數(shù)器,當前計數(shù)器初始值為5。

tryAcquireShared方法其實就是判斷一下當前計數(shù)器的值,是否為0了,如果為0的話返回1(返回1的時候,就表示獲取鎖成功,awit()方法就不再阻塞)。

tryReleaseShared方法就是利用CAS的方式,對計數(shù)器進行減一的操作,而我們實際上每次調(diào)用countDownLatch.countDown()方法的時候,最終都會調(diào)到這個方法,對計數(shù)器進行減一操作,一直減到0為止。

countDownLatch.await()

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

代碼很簡單,就一句話(注意acquireSharedInterruptibly()方法是抽象類:AbstractQueuedSynchronizer的一個方法,我們上面提到的Sync繼承了它),我們跟蹤源碼,繼續(xù)往下看:

acquireSharedInterruptibly(int arg) 
public final void acquireSharedInterruptibly(int arg)
throws InterruptedException {
if (Thread.interrupted())
throw new InterruptedException();
if (tryAcquireShared(arg) < 0)
doAcquireSharedInterruptibly(arg);
}

源碼也是非常簡單的,首先判斷了一下,當前線程是否有被中斷,如果沒有的話,就調(diào)用tryAcquireShared(int acquires)方法,判斷一下當前線程是否還需要“阻塞”。其實這里調(diào)用的tryAcquireShared方法,就是我們上面提到的java.util.concurrent.CountDownLatch.Sync.tryAcquireShared(int)這個方法。

當然,在一開始我們沒有調(diào)用過countDownLatch.countDown()方法時,這里tryAcquireShared方法肯定是會返回-1的,因為會進入到doAcquireSharedInterruptibly方法。 

doAcquireSharedInterruptibly(int arg)

countDown()方法

// 計數(shù)器減1
public void countDown() {
sync.releaseShared(1); 
}
//調(diào)用AQS的releaseShared方法
public final boolean releaseShared(int arg) {
if (tryReleaseShared(arg)) {//計數(shù)器減一
doReleaseShared();//喚醒后繼結(jié)點,這個時候隊列中可能只有調(diào)用過await()的線程節(jié)點,也可能隊列為空
return true;
}
return false;
}

這個時候,我們應該對于countDownLatch.await()方法是怎么“阻塞”當前線程的,已經(jīng)非常明白了。其實說白了,就是當你調(diào)用了countDownLatch.await()方法后,你當前線程就會進入了一個死循環(huán)當中,在這個死循環(huán)里面,會不斷的進行判斷,通過調(diào)用tryAcquireShared方法,不斷判斷我們上面說的那個計數(shù)器,看看它的值是否為0了(為0的時候,其實就是我們調(diào)用了足夠多 countDownLatch.countDown()方法的時候),如果是為0的話,tryAcquireShared就會返回1,代碼也會進入到圖中的紅框部分,然后跳出了循環(huán),也就不再“阻塞”當前線程了。

需要注意的是,說是在不停的循環(huán),其實也并非在不停的執(zhí)行for循環(huán)里面的內(nèi)容,因為在后面調(diào)用parkAndCheckInterrupt()方法時,在這個方法里面是會調(diào)用 LockSupport.park(this);來掛起當前線程。

CountDownLatch 使用的注意點:

1、只有當count為0時,await之后的程序才夠執(zhí)行。

2、countDown必須寫在finally中,防止發(fā)生異程常時,導致程序死鎖。

使用場景:

比如對于馬拉松比賽,進行排名計算,參賽者的排名,肯定是跑完比賽之后,進行計算得出的,翻譯成Java識別的預發(fā),就是N個線程執(zhí)行操作,主線程等到N個子線程執(zhí)行完畢之后,在繼續(xù)往下執(zhí)行。

public static void testCountDownLatch(){
int threadCount = 10;
final CountDownLatch latch = new CountDownLatch(threadCount);
for(int i=0; i< threadCount; i++){
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("線程" + Thread.currentThread().getId() + "開始出發(fā)");
try {
Thread.sleep(1000);
System.out.println("線程" + Thread.currentThread().getId() + "已到達終點");
} catch (InterruptedException e) {
e.printStackTrace();
} fianlly {
latch.countDown();
}
}
}).start();
}
try {
latch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("10個線程已經(jīng)執(zhí)行完畢!開始計算排名");
}

結(jié)果:

線程10開始出發(fā)
線程13開始出發(fā)
線程12開始出發(fā)
線程11開始出發(fā)
線程14開始出發(fā)
線程15開始出發(fā)
線程16開始出發(fā)
線程17開始出發(fā)
線程18開始出發(fā)
線程19開始出發(fā)
線程14已到達終點
線程15已到達終點
線程13已到達終點
線程12已到達終點
線程10已到達終點
線程11已到達終點
線程16已到達終點
線程17已到達終點
線程18已到達終點
線程19已到達終點
10個線程已經(jīng)執(zhí)行完畢!開始計算排名

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java 將一個字符重復n遍過程詳解

    Java 將一個字符重復n遍過程詳解

    這篇文章主要介紹了Java 將一個字符重復n遍過程詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-10-10
  • springboot2.5.2與 flowable6.6.0整合流程引擎應用分析

    springboot2.5.2與 flowable6.6.0整合流程引擎應用分析

    這篇文章主要介紹了springboot2.5.2與 flowable6.6.0整合流程引擎應用分析,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-07-07
  • Javassist用法詳解

    Javassist用法詳解

    這篇文章主要介紹了Javassist用法的相關(guān)資料,幫助大家更好的理解和學習使用Java,感興趣的朋友可以了解下
    2021-02-02
  • 最新版Spring Security中的路徑匹配方案

    最新版Spring Security中的路徑匹配方案

    在 Spring Security 中,路徑匹配是權(quán)限控制的核心部分,它決定了哪些請求可以訪問特定的資源,本文將詳細介紹 Spring Security 中的路徑匹配策略,并提供相應的代碼示例,需要的朋友可以參考下
    2024-04-04
  • Java實現(xiàn)去掉字符串重復字母的方法示例

    Java實現(xiàn)去掉字符串重復字母的方法示例

    這篇文章主要介紹了Java實現(xiàn)去掉字符串重復字母的方法,涉及java針對字符串的遍歷、判斷、運算等相關(guān)操作技巧,需要的朋友可以參考下
    2017-12-12
  • java并發(fā)編程專題(十)----(JUC原子類)基本類型詳解

    java并發(fā)編程專題(十)----(JUC原子類)基本類型詳解

    這篇文章主要介紹了java JUC原子類基本類型詳解的相關(guān)資料,文中示例代碼非常詳細,幫助大家更好的理解和學習,感興趣的朋友可以了解下
    2020-07-07
  • Java的類型擦除式泛型詳解

    Java的類型擦除式泛型詳解

    Java語言中的泛型只存在于程序源碼之中,在編譯后的字節(jié)碼文件里,則全部泛型都會被替換為原來的原始類型(Raw Type),并且會在相應的地方插入強制轉(zhuǎn)型的代碼。這篇文章主要介紹了Java的類型擦除式泛型,需要的朋友可以參考下
    2021-08-08
  • MyBatis通用Mapper和PageHelper的過程詳解

    MyBatis通用Mapper和PageHelper的過程詳解

    這篇文章主要介紹了MyBatis通用Mapper和PageHelper的相關(guān)知識,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-11-11
  • http協(xié)議進階之Transfer-Encoding和HttpCore實現(xiàn)詳解

    http協(xié)議進階之Transfer-Encoding和HttpCore實現(xiàn)詳解

    這篇文章主要給大家介紹了http協(xié)議之Transfer-Encoding和HttpCore實現(xiàn)的相關(guān)資料,文中介紹的非常詳細,相信對大家具有一定的參考價值,需要的朋友們下面來一起看看吧。
    2017-04-04
  • Java 互相關(guān)聯(lián)的實體無限遞歸問題的解決

    Java 互相關(guān)聯(lián)的實體無限遞歸問題的解決

    這篇文章主要介紹了Java 互相關(guān)聯(lián)的實體無限遞歸問題的解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10

最新評論