Java中的CAS和自旋鎖詳解
什么是CAS
CAS算法(Compare And Swap),即比較并替換,是一種實現(xiàn)并發(fā)編程時常用到的算法,Java并發(fā)包中的很多類都使用了CAS算法。
CAS算法有3個基本操作數(shù):
- 內(nèi)存地址V
- 舊的預(yù)期值A(chǔ)
- 要修改的新值B
CAS使用自旋的方式來交換值,操作步驟為:
- 讀取內(nèi)存地址V的值保存在A中
- 在原子操作中比較內(nèi)存地址V的值是否與A相同
- 相同時,修改內(nèi)存地址V的值為B,原子操作成功。
- 不相同時,循環(huán)執(zhí)行第一至第三步(自旋),直到成功。
什么是自旋鎖?
自旋鎖(spinlock):是指當(dāng)一個線程在獲取鎖的時候,如果鎖已經(jīng)被其它線程獲取,那么該線程將循環(huán)等待,然后不斷的判斷鎖是否能夠被成功獲取,直到獲取到鎖才會退出循環(huán)。
自旋鎖與互斥鎖比較類似,它們都是為了解決對某項資源的互斥使用。無論是互斥鎖,還是自旋鎖,在任何時刻,最多只能有一個保持者,也就說,在任何時刻最多只能有一個執(zhí)行單元獲得鎖。
對于互斥鎖,會讓沒有得到鎖資源的線程進入BLOCKED狀態(tài),而后在爭奪到鎖資源后恢復(fù)為RUNNABLE狀態(tài),這個過程中涉及到操作系統(tǒng)用戶模式和內(nèi)核模式的轉(zhuǎn)換,代價比較高。但是自旋鎖不會引起調(diào)用者堵塞,如果自旋鎖已經(jīng)被別的執(zhí)行單元保持,調(diào)用者就一直循環(huán)在那里看是否該自旋鎖的保持者已經(jīng)釋放了鎖。
自旋鎖的實現(xiàn)基礎(chǔ)是CAS算法機制。CAS自旋鎖屬于樂觀鎖,樂觀地認(rèn)為程序中的并發(fā)情況不那么嚴(yán)重,所以讓線程不斷去嘗試更新。
基于CAS實現(xiàn)-原子操作類
先看一個線程不安全的例子:
public class AutomicDemo {
public static int num = 0;
public static void main(String[] args){
for(int i=0; i<5; i++){
new Thread(new Runnable() {
@Override
public void run() {
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
for(int j=0; j<200; j++){
num++;
}
}
}).start();
}
/* 主控失眠3秒,保證所有線程執(zhí)行完成 */
try {
Thread.sleep(15000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("num=" + num);
}
}輸出結(jié)果:
num=950
因為自增操作不是原子性,多線程環(huán)境下,訪問共享變量線程不安全。
解決方法,加synchronized同步鎖:
for(int j=0; j<200; j++){
synchronized (AutomicDemo.class){
num++;
}
}輸出結(jié)果:
num=1000
線程安全。
synchronized確保了線程安全,但會讓沒有得到鎖資源的線程進入BLOCKED狀態(tài),而后在爭奪到鎖資源后恢復(fù)為RUNNABLE狀態(tài),這個過程中涉及到操作系統(tǒng)用戶模式和內(nèi)核模式的轉(zhuǎn)換,代價比較高。
盡管Java1.6為Synchronized做了優(yōu)化,增加了從偏向鎖到輕量級鎖再到重量級鎖的過度,但是在最終轉(zhuǎn)變?yōu)橹亓考夋i之后,性能仍然較低。
于是JDK提供了一系列原子操作類:AtomicInteger、AtomicLong、AtomicBoolean、AtomicReference等,它們都是基于CAS去實現(xiàn)的,下面我們就來詳細(xì)看一看原子操作類。
public class AutomicDemo {
public static AtomicInteger num = new AtomicInteger(0);
public static void main(String[] args){
for(int i=0; i<5; i++){
new Thread(new Runnable() {
@Override
public void run() {
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
for(int j=0; j<200; j++){
num.incrementAndGet();
}
}
}).start();
}
/* 主控失眠3秒,保證所有線程執(zhí)行完成 */
try {
Thread.sleep(15000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("num=" + num.get());
}
}輸出:
num=1000
使用AtomicInteger之后,最終的輸出結(jié)果同樣可以保證是200。并且在某些情況下,代碼的性能會比Synchronized更好。
原子類操作方法
- public final int get():取得當(dāng)前值
- public final void set(int newValue):設(shè)置當(dāng)前值
- public final int getAndSet(int newValue):設(shè)置新值并返回舊值
- public final boolean compareAndSet(int expect, int update):如果當(dāng)前值為expect,則設(shè)置為update
- public final boolean weakCompareAndSet(int expect, int update):如果當(dāng)前值為expect,則設(shè)置為update,可能失敗,不提供保障
- public final int getAndIncrement():當(dāng)前值加1,返回舊值
- public final int getAndDecrement():當(dāng)前值減1,返回舊值
- public final int getAndAdd(int delta):當(dāng)前值加delta,返回舊值
- public final int incrementAndGet():當(dāng)前值加1,返回新值
- public final int decrementAndGet():當(dāng)前值減1,返回新值
- public final int addAndGet(int delta):當(dāng)前值加delta,返回新值
AtomicInteger底層原理
所有Atomic相關(guān)類的實現(xiàn)都是通過CAS(Compare And Swap)去實現(xiàn)的,它是一種樂觀鎖的實現(xiàn)。
CAS實現(xiàn)放在 Unsafe 這個類中的,其內(nèi)部大部分是native方法。這個類是不允許更改的,而且也不建議開發(fā)者調(diào)用,它只是用于JDK內(nèi)部調(diào)用,看名字就知道它是不安全的,因為它是直接操作內(nèi)存,稍不注意就可能把內(nèi)存寫崩。
Unsafe類使Java擁有了像C語言的指針一樣操作內(nèi)存空間的能力。有一下特點:
1、不受jvm管理,也就意味著無法被GC,需要我們手動GC,稍有不慎就會出現(xiàn)內(nèi)存泄漏。
2、Unsafe的不少方法中必須提供原始地址(內(nèi)存地址)和被替換對象的地址,偏移量要自己計算,一旦出現(xiàn)問題就是JVM崩潰級別的異常,會導(dǎo)致整個JVM實例崩潰,表現(xiàn)為應(yīng)用程序直接crash掉。
3、直接操作內(nèi)存,也意味著其速度更快,在高并發(fā)的條件之下能夠很好地提高效率。
去看AtomicInteger的內(nèi)部實現(xiàn)可以發(fā)現(xiàn),全是調(diào)用的Unsafe類中的方法:

CAS機制ABA問題
ABA問題:CAS算法通過比較變量值是否相同來修改變量值以保證原子性,但如果一個內(nèi)存地址的值本身是A,線程1準(zhǔn)備修改為C。在這期間,線程2將值修改為B,線程3將值修改為A,線程1獲取內(nèi)存地址的值還是A,故修改為C成功。但獲取的A已不再是最開始那一個A。這就是經(jīng)典的ABA問題,A已不再是A。
如何解決ABA問題呢?
兩個方法,1、增加版本號;2、增加時間戳。
- 增加版本號:讓值的修改從A-B-A-C變?yōu)?A-2B-3A-4C;這樣在線程1 中就能判別出1A不是當(dāng)前內(nèi)存中的3A,從而不會更新變量為4C。
- 增加時間戳:值被修改時,除了更新數(shù)據(jù)本身外,還必須更新時間戳。對象值以及時間戳都必須滿足期望,寫入才會成功。JDK提供了一個帶有時間戳的CAS操作類AtomicStampedeReference。
CAS的缺點
Java原子類使用自旋的方式來處理每次比較不相同時后的重試操作,下面來看看AtomicInteger類incrementAndGet方法的代碼:
//AtomicInteger 的incrementAndGet方法,變量U為靜態(tài)常量jdk.internal.misc.Unsafe類型
public final int incrementAndGet() {
//使用getAndAddInt方法,實際操作類似j++
return U.getAndAddInt(this, VALUE, 1) + 1;
}
//jdk.internal.misc.Unsafe類型的getAndAddInt方法
public final int getAndAddInt(Object o, long offset, int delta) {
int v;
do {
//獲取變量o的可見值
v = getIntVolatile(o, offset);
//比較與替換變量o(CAS算法)
} while (!weakCompareAndSetInt(o, offset, v, v + delta));
return v;
}
//jdk.internal.misc.Unsafe類型的weakCompareAndSetInt方法
public final boolean weakCompareAndSetInt(Object o, long offset,
int expected,
int x) {
//執(zhí)行比較與替換
return compareAndSetInt(o, offset, expected, x);
}在Unsafe類的getAndAddInt方法中使用了do…while循環(huán)語句,循環(huán)語句的作用為自旋,Unsafe的weakCompareAndSetInt實現(xiàn)CAS算法。
如果weakCompareAndSetInt一直不成功將會一直自旋下去,這將消耗過多的CPU時間。而且原子類使用CAS算法實現(xiàn),這導(dǎo)致原子類只能保證一個變量的原子操作,對于需要保證一個具有多個操作的事務(wù)將變得無能為力。
總結(jié)如下:
- CPU開銷較大
- 在并發(fā)量比較高的情況下,如果許多線程反復(fù)嘗試更新某一個變量,卻又一直更新不成功,循環(huán)往復(fù),自旋會給CPU帶來很大的壓力。
- 不能保證代碼塊的原子性
- CAS機制所保證的只是一個變量的原子性操作,而不能保證整個代碼塊的原子性。比如需要保證3個變量共同進行原子性的更新,就不得不使用Synchronized了。
為應(yīng)對CAS存在缺點,替換方案如下:
- 自旋效率:Java提供了自適應(yīng)自旋鎖
- 片面性:應(yīng)對片面性問題Java提供了讀寫鎖
- ABA問題:用AtomicStampedReference/AtomicMarkableReference解決ABA問題
到此這篇關(guān)于Java中的CAS和自旋鎖詳解的文章就介紹到這了,更多相關(guān)Java的CAS和自旋鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring aop action中驗證用戶登錄狀態(tài)的實例代碼
本篇文章主要介紹了spring aop action中驗證用戶登錄狀態(tài)的實例代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07
mybatis?返回Map類型key默認(rèn)為大寫問題
這篇文章主要介紹了mybatis?返回Map類型key默認(rèn)為大寫問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11
關(guān)于thymeleaf判斷對象是否為空的相關(guān)邏輯處理
這篇文章主要介紹了關(guān)于thymeleaf判斷對象是否為空的相關(guān)邏輯處理,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-10-10
springboot整合redis進行數(shù)據(jù)操作(推薦)
springboot整合redis比較簡單,并且使用redistemplate可以讓我們更加方便的對數(shù)據(jù)進行操作。下面通過本文給大家分享springboot整合redis進行數(shù)據(jù)操作的相關(guān)知識,感興趣的朋友一起看看吧2017-10-10
AJAX Servlet實現(xiàn)數(shù)據(jù)異步交互的方法
本篇文章主要介紹了AJAX Servlet實現(xiàn)數(shù)據(jù)異步交互的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-07-07
SpringBoot中Formatter和Converter用法和區(qū)別小結(jié)
本文主要介紹了SpringBoot中Formatter和Converter用法和區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07

