亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

學(xué)習(xí)非阻塞的同步機(jī)制CAS

 更新時間:2019年05月31日 16:01:41   作者:展翅而飛  
現(xiàn)代的處理器都包含對并發(fā)的支持,其中最通用的方法就是比較并交換(compare and swap),簡稱CAS。下面我們來一起學(xué)習(xí)一下吧

在研究線程池的執(zhí)行原理時,看到一段不斷循環(huán)重試的代碼,不理解它的原理,看注釋這是CAS的實現(xiàn),所以學(xué)會之后記錄下來。

鎖有什么劣勢

在多線程并發(fā)下,可以通過加鎖來保證線程安全性,但多個線程同時請求鎖,很多情況下避免不了要借助操作系統(tǒng),線程掛起和恢復(fù)會存在很大的開銷,并存在很長時間的中斷。一些細(xì)粒度的操作,例如同步容器,操作往往只有很少代碼量,如果存在鎖并且線程激烈地競爭,調(diào)度的代價很大。
總結(jié)來說,線程持有鎖,會讓其他需要鎖的線程阻塞,產(chǎn)生多種風(fēng)險和開銷。加鎖是一種悲觀方法,線程總是設(shè)想在自己持有資源的同時,肯定有其他線程想要資源,不牢牢鎖住資源還不能放心呢。
在硬件的支持下,出現(xiàn)了非阻塞的同步機(jī)制,其中一種常用實現(xiàn)就是CAS。

什么是CAS

現(xiàn)代的處理器都包含對并發(fā)的支持,其中最通用的方法就是比較并交換(compare and swap),簡稱CAS。

CAS 操作包含三個操作數(shù) —— 內(nèi)存位置(V)、預(yù)期原值(A)和新值(B)。如果內(nèi)存位置的值與預(yù)期原值相匹配,那么處理器會自動將該位置值更新為新值。否則,處理器不做任何操作。無論V值是否等于A值,都將返回V的原值。CAS 有效地說明了:我認(rèn)為位置 V 應(yīng)該包含值 A;如果包含該值,則將 B 放到這個位置;否則,不要更改該位置,只告訴我這個位置現(xiàn)在的值即可。

當(dāng)多個線程嘗試使用CAS同時更新一個變量,最終只有一個線程會成功,其他線程都會失敗。但和使用鎖不同,失敗的線程不會被阻塞,而是被告之本次更新操作失敗了,可以再試一次。此時,線程可以根據(jù)實際情況,繼續(xù)重試或者跳過操作,大大減少因為阻塞而損失的性能。所以,CAS是一種樂觀的操作,它希望每次都能成功地執(zhí)行更新操作。

public class SimulationCAS {
private int value;
public synchronized int get() {
return value;
}
public synchronized boolean compareAndSet(int expectedValue, int newValue) {
if (expectedValue == compareAndSwap(expectedValue, newValue)) {
return true;
}
return false;
}
public synchronized int compareAndSwap(int expectedValue, int newValue) {
int oldValue = value;
if (oldValue == expectedValue) {
value = newValue;
}
return oldValue;
}
}

上面的代碼模擬了CAS的操作,其中compareAndSwap是CAS語義的體現(xiàn),compareAndSet對value進(jìn)行了更新操作,并返回成功與否。
幾行代碼就實現(xiàn)了CAS,是不是覺得很簡單呢?但你要知道,CAS僅僅告訴你操作結(jié)果,操作失敗后一系列重試回退放棄等操作都要自己實現(xiàn),開發(fā)起來遠(yuǎn)比使用鎖復(fù)雜。

Atom原子類

JVM是支持CAS的,體現(xiàn)在我們常用的Atom原子類,拿AtomicInteger分析一下源碼。

public final int getAndIncrement() {
for (;;) {
int current = get();
int next = current + 1;
if (compareAndSet(current, next))
return current;
}
}

對AtomicInteger進(jìn)行+1操作,循環(huán)里,會將當(dāng)前值和+1后的目標(biāo)值傳入compareAndSet,直到成功才跳出方法。compareAndSet是不是很熟悉呢,接著來看看它的代碼。

// setup to use Unsafe.compareAndSwapInt for updates
private static final Unsafe unsafe = Unsafe.getUnsafe();
public final boolean compareAndSet(int expect, int update) {
return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
}

compareAndSet調(diào)用了unsafe.compareAndSwapInt,這是一個native方法,原理就是調(diào)用硬件支持的CAS方法。看懂這個應(yīng)該就能明白Atom類的原理,其他方法的實現(xiàn)是類似的。

線程池里的CAS

有了CAS的基礎(chǔ)后,可以來研究那段我未看懂的代碼。
提交一個執(zhí)行任務(wù),線程池會嘗試增加一個工作線程去處理任務(wù)。下面是ThreadPoolExecutor里addWorker的一段代碼:

private boolean addWorker(Runnable firstTask, boolean core) {
retry:
for (;;) {
int c = ctl.get();
int rs = runStateOf(c);
// Check if queue empty only if necessary.
if (rs >= SHUTDOWN &&
! (rs == SHUTDOWN &&
firstTask == null &&
! workQueue.isEmpty()))
return false;
for (;;) {
int wc = workerCountOf(c);
if (wc >= CAPACITY ||
wc >= (core ? corePoolSize : maximumPoolSize))
return false;
if (compareAndIncrementWorkerCount(c))
break retry;
c = ctl.get(); // Re-read ctl
if (runStateOf(c) != rs)
continue retry;
// else CAS failed due to workerCount change; retry inner loop
}
}

//其他省略

在內(nèi)循環(huán)里,會調(diào)用compareAndIncrementWorkerCount方法增加一個工作線程,原理和AtomicInteger的getAndIncrement方法是一樣的。如果增加成功,直接跳出循環(huán),否則在檢查線程池狀態(tài)后,再次在內(nèi)循環(huán)調(diào)用compareAndIncrementWorkerCount,直到添加成功。

現(xiàn)在再看代碼,瞬間就明白了。

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

相關(guān)文章

  • Java中classpath講解及使用方式

    Java中classpath講解及使用方式

    本文詳細(xì)講解了Java中classpath講解及使用方式,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-12-12
  • Spring自動配置之condition條件判斷下篇

    Spring自動配置之condition條件判斷下篇

    這篇文章主要為大家介紹了SpringBoot?condition條件判斷功能的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-08-08
  • Spring使用注解存儲和讀取對象詳解

    Spring使用注解存儲和讀取對象詳解

    這篇文章主要給大家介紹了關(guān)于Spring如何通過注解存儲和讀取對象的相關(guān)資料,文中通過實例代碼介紹的非常詳細(xì),有一定的參考價值,需要的朋友可以參考下
    2023-04-04
  • 改善Java代碼之慎用java動態(tài)編譯

    改善Java代碼之慎用java動態(tài)編譯

    這篇文章主要介紹了改善Java代碼之慎用java動態(tài)編譯,需要的朋友可以參考下
    2021-04-04
  • Java concurrency之集合_動力節(jié)點Java學(xué)院整理

    Java concurrency之集合_動力節(jié)點Java學(xué)院整理

    Java集合主體內(nèi)容包括Collection集合和Map類;而Collection集合又可以劃分為List(隊列)和Set(集合),有需要的小伙伴可以參考下
    2017-06-06
  • Mapper層繼承BaseMapper<T>需要引入的pom依賴方式

    Mapper層繼承BaseMapper<T>需要引入的pom依賴方式

    這篇文章主要介紹了Mapper層繼承BaseMapper<T>需要引入的pom依賴方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • log4j的配置文件詳細(xì)解析

    log4j的配置文件詳細(xì)解析

    以下小編主要為大家介紹一下log4j的配置文件各個配置項的含義。需要的朋友可以過來參考下
    2013-08-08
  • Java8之Lambda表達(dá)式使用解讀

    Java8之Lambda表達(dá)式使用解讀

    這篇文章主要介紹了Java8之Lambda表達(dá)式使用解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-11-11
  • 使用Java實現(xiàn)簡單搭建內(nèi)網(wǎng)穿透

    使用Java實現(xiàn)簡單搭建內(nèi)網(wǎng)穿透

    內(nèi)網(wǎng)穿透是一種網(wǎng)絡(luò)技術(shù),適用于需要遠(yuǎn)程訪問本地部署服務(wù)的場景,本文主要為大家介紹了如何使用Java實現(xiàn)簡單搭建內(nèi)網(wǎng)穿透,感興趣的可以了解下
    2024-02-02
  • 我賭你不清楚Spring中關(guān)于Null的這些事

    我賭你不清楚Spring中關(guān)于Null的這些事

    這篇文章主要介紹了我賭你不清楚Spring中關(guān)于Null的這些事,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-06-06

最新評論