中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

非阻塞的同步機制CAS怎么用

發布時間:2021-08-21 14:20:05 來源:億速云 閱讀:130 作者:小新 欄目:編程語言

這篇文章給大家分享的是有關非阻塞的同步機制CAS怎么用的內容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。

鎖有什么劣勢

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

什么是CAS

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

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

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

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語義的體現,compareAndSet對value進行了更新操作,并返回成功與否。
幾行代碼就實現了CAS,是不是覺得很簡單呢?但你要知道,CAS僅僅告訴你操作結果,操作失敗后一系列重試回退放棄等操作都要自己實現,開發起來遠比使用鎖復雜。

Atom原子類

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

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

對AtomicInteger進行+1操作,循環里,會將當前值和+1后的目標值傳入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調用了unsafe.compareAndSwapInt,這是一個native方法,原理就是調用硬件支持的CAS方法。看懂這個應該就能明白Atom類的原理,其他方法的實現是類似的。

線程池里的CAS

有了CAS的基礎后,可以來研究那段我未看懂的代碼。
提交一個執行任務,線程池會嘗試增加一個工作線程去處理任務。下面是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
}
}

//其他省略

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

感謝各位的閱讀!關于“非阻塞的同步機制CAS怎么用”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

cas
AI

延安市| 曲阳县| 安吉县| 读书| 沾益县| 松滋市| 辽阳市| 庄浪县| 乐陵市| 香河县| 阳东县| 千阳县| 万载县| 潮安县| 巴马| 岢岚县| 望奎县| 黄陵县| 宁蒗| 四子王旗| 桂东县| 德格县| 江津市| 乐山市| 新密市| 永春县| 赤峰市| 萨迦县| 汕头市| 重庆市| 上杭县| 苗栗市| 浦城县| 临夏市| 石楼县| 长泰县| 通海县| 和顺县| 铜陵市| 衡山县| 桂平市|