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

Java CAS底層實(shí)現(xiàn)原理實(shí)例詳解

 更新時(shí)間:2020年01月10日 11:07:36   作者:VincentYew  
這篇文章主要介紹了Java CAS底層實(shí)現(xiàn)原理實(shí)例詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

這篇文章主要介紹了Java CAS底層實(shí)現(xiàn)原理實(shí)例詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

一、CAS(compareAndSwap)的概念

CAS,全稱Compare And Swap(比較與交換),解決多線程并行情況下使用鎖造成性能損耗的一種機(jī)制。

CAS(V, A, B),V為內(nèi)存地址、A為預(yù)期原值,B為新值。如果內(nèi)存地址的值與預(yù)期原值相匹配,那么將該位置值更新為新值。否則,說(shuō)明已經(jīng)被其他線程更新,處理器不做任何操作;無(wú)論哪種情況,它都會(huì)在 CAS 指令之前返回該位置的值。而我們可以使用自旋鎖,循環(huán)CAS,重新讀取該變量再嘗試再次修改該變量,也可以放棄操作。

二、CAS(compareAndSwap)的產(chǎn)生

為什么需要CAS機(jī)制呢?我們先從一個(gè)錯(cuò)誤現(xiàn)象談起。我們經(jīng)常使用volatile關(guān)鍵字修飾某一個(gè)變量,表明這個(gè)變量是全局共享的一個(gè)變量,同時(shí)具有了可見(jiàn)性和有序性。但是卻沒(méi)有原子性。比如說(shuō)一個(gè)常見(jiàn)的操作a++。這個(gè)操作其實(shí)可以細(xì)分成三個(gè)步驟:

(1)從內(nèi)存中讀取a

(2)對(duì)a進(jìn)行加1操作

(3)將a的值重新寫(xiě)入內(nèi)存中

在單線程狀態(tài)下這個(gè)操作沒(méi)有一點(diǎn)問(wèn)題,但是在多線程中就會(huì)出現(xiàn)各種各樣的問(wèn)題了。因?yàn)榭赡芤粋€(gè)線程對(duì)a進(jìn)行了加1操作,還沒(méi)來(lái)得及寫(xiě)入內(nèi)存,其他的線程就讀取了舊值。造成了線程的不安全現(xiàn)象。

Volatile關(guān)鍵字可以保證線程間對(duì)于共享變量的可見(jiàn)性可有序性,可以防止CPU的指令重排序(DCL單例),但是無(wú)法保證操作的原子性,所以jdk1.5之后引入CAS利用CPU原語(yǔ)保證線程操作的院子性。

CAS操作由處理器提供支持,是一種原語(yǔ)。原語(yǔ)是操作系統(tǒng)或計(jì)算機(jī)網(wǎng)絡(luò)用語(yǔ)范疇。是由若干條指令組成的,用于完成一定功能的一個(gè)過(guò)程,具有不可分割性,即原語(yǔ)的執(zhí)行必須是連續(xù)的,在執(zhí)行過(guò)程中不允許被中斷。如 Intel 處理器,比較并交換通過(guò)指令的 cmpxchg 系列實(shí)現(xiàn)。

三、CAS(compareAndSwap)的原理探究

CAS的實(shí)現(xiàn)主要在JUC中的atomic包,我們以AtomicInteger類為例:

通過(guò)代碼追溯,可以看出JAVA中的CAS操作都是通過(guò)sun包下Unsafe類實(shí)現(xiàn),而Unsafe類中的方法都是native方法,由JVM本地實(shí)現(xiàn),所以最終的實(shí)現(xiàn)是基于C、C++在操作系統(tǒng)之上操作

Unsafe類,在sun.misc包下,不屬于Java標(biāo)準(zhǔn)。Unsafe類提供一系列增加Java語(yǔ)言能力的操作,如內(nèi)存管理、操作類/對(duì)象/變量、多線程同步等

//var1為CAS操作的對(duì)象,offset為var1某個(gè)屬性的地址偏移值,expected為期望值,var2為要設(shè)置的值,利用JNI來(lái)完成CPU指令的操作
public final native boolean compareAndSwapObject(Object var1, long var2, Object var4, Object var5);
public final native boolean compareAndSwapInt(Object var1, long var2, int var4, int var5);
public final native boolean compareAndSwapLong(Object var1, long var2, long var4, long var6);
public native Object getObjectVolatile(Object var1, long var2);
public native void putObjectVolatile(Object var1, long var2, Object var4);
Hotspot源碼中關(guān)于unsafe的實(shí)現(xiàn)hotspot\src\share\vm\prims\unsafe.cpp
UNSAFE_ENTRY(jboolean, Unsafe_CompareAndSwapInt(JNIEnv *env, jobject unsafe, jobject obj, jlong offset, jint e, jint x))
 UnsafeWrapper("Unsafe_CompareAndSwapInt");
 oop p = JNIHandles::resolve(obj);根據(jù)偏移量,計(jì)算value的地址。這里的offset就是 AtomaicInteger中的valueOffset
 jint* addr = (jint *) index_oop_from_field_offset_long(p, offset);
 return (jint)(Atomic::cmpxchg(x, addr, e)) == e;
UNSAFE_END\hotspot\src\share\vm\runtime\atomic.cppunsigned Atomic::cmpxchg(unsigned int exchange_value,
             volatile unsigned int* dest, unsigned int compare_value) {
 assert(sizeof(unsigned int) == sizeof(jint), "more work to do");
 return (unsigned int)Atomic::cmpxchg((jint)exchange_value, (volatile jint*)dest,
                    (jint)compare_value);
}根據(jù)操作系統(tǒng)類型調(diào)用不同平臺(tái)下的重載函數(shù),這個(gè)在預(yù)編譯期間編譯器會(huì)決定調(diào)用哪個(gè)平臺(tái)下的重載

可以看到調(diào)用了“Atomic::cmpxchg”方法,“Atomic::cmpxchg”方法在linux_x86和windows_x86的實(shí)現(xiàn)如下

linux_x86底層實(shí)現(xiàn)\hotspot\src\os_cpu\linux_x86\vm\atomic_linux_x86.inline.hpp
inline jint   Atomic::cmpxchg  (jint   exchange_value, volatile jint*   dest, jint   compare_value) {
 int mp = os::is_MP();
 __asm__ volatile (LOCK_IF_MP(%4) "cmpxchgl %1,(%3)"
          : "=a" (exchange_value)
          : "r" (exchange_value), "a" (compare_value), "r" (dest), "r" (mp)
          : "cc", "memory");
 return exchange_value;
}
windows_x86底層實(shí)現(xiàn)
hotspot\src\os_cpu\windows_x86\vmatomic_linux_x86.inline.hpp
inline jint   Atomic::cmpxchg  (jint   exchange_value, volatile jint*   dest, jint   compare_value) {
 // alternative for InterlockedCompareExchange
 int mp = os::is_MP();
 __asm {
  mov edx, dest
  mov ecx, exchange_value
  mov eax, compare_value
  LOCK_IF_MP(mp)
  cmpxchg dword ptr [edx], ecx
 }
}

總結(jié):根據(jù)資料查詢,其實(shí)CAS底層實(shí)現(xiàn)根據(jù)不同的操作系統(tǒng)會(huì)有不同重載,CAS的實(shí)現(xiàn)離不開(kāi)處理器的支持。

核心代碼就是一條帶lock 前綴的 cmpxchg 指令,即lock cmpxchg dword ptr [edx], ecx

Atomic::cmpxchg方法解析:

mp是“os::is_MP()”的返回結(jié)果,“os::is_MP()”是一個(gè)內(nèi)聯(lián)函數(shù),用來(lái)判斷當(dāng)前系統(tǒng)是否為多處理器。

如果當(dāng)前系統(tǒng)是多處理器,該函數(shù)返回1。

否則,返回0。

LOCK_IF_MP(mp)會(huì)根據(jù)mp的值來(lái)決定是否為cmpxchg指令添加lock前綴。

如果通過(guò)mp判斷當(dāng)前系統(tǒng)是多處理器(即mp值為1),則為cmpxchg指令添加lock前綴。

否則,不加lock前綴。

這是一種優(yōu)化手段,認(rèn)為單處理器的環(huán)境沒(méi)有必要添加lock前綴,只有在多核情況下才會(huì)添加lock前綴,因?yàn)閘ock會(huì)導(dǎo)致性能下降。cmpxchg是匯編指令,作用是比較并交換操作數(shù)。

四、CAS機(jī)制的優(yōu)缺點(diǎn)

4.1 優(yōu)點(diǎn)

cas是一種樂(lè)觀鎖,而且是一種非阻塞的輕量級(jí)的樂(lè)觀鎖,什么是非阻塞式的呢?其實(shí)就是一個(gè)線程想要獲得鎖,對(duì)方會(huì)給一個(gè)回應(yīng)表示這個(gè)鎖能不能獲得。在資源競(jìng)爭(zhēng)不激烈的情況下性能高,相比synchronized重量鎖,synchronized會(huì)進(jìn)行比較復(fù)雜的加鎖,解鎖和喚醒操作。

4.2 缺點(diǎn)

1)循環(huán)時(shí)間長(zhǎng)開(kāi)銷大,占用CPU資源

2)只能保證一個(gè)共享變量的原子操作

3)ABA問(wèn)題

4.3 解決ABA問(wèn)題

1)添加版本號(hào)

2)AtomicStampedReference

java并發(fā)包為了解決這個(gè)問(wèn)題,提供了一個(gè)帶有標(biāo)記的原子引用類“AtomicStampedReference”,它可以通過(guò)控制變量值的版本來(lái)保證CAS的正確性。因此,在使用CAS前要考慮清楚“ABA”問(wèn)題是否會(huì)影響程序并發(fā)的正確性,如果需要解決ABA問(wèn)題,改用傳統(tǒng)的互斥同步可能會(huì)比原子類更高效。

五、CAS使用的時(shí)機(jī)

5.1 線程數(shù)較少、等待時(shí)間短可以采用自旋鎖進(jìn)行CAS嘗試拿鎖,較于synchronized高效

5.2 線程數(shù)較大、等待時(shí)間長(zhǎng),不建議使用自旋鎖,占用CPU較高

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

相關(guān)文章

  • java用接口、多態(tài)、繼承、類計(jì)算三角形和矩形周長(zhǎng)及面積的方法

    java用接口、多態(tài)、繼承、類計(jì)算三角形和矩形周長(zhǎng)及面積的方法

    這篇文章主要介紹了java用接口、多態(tài)、繼承、類計(jì)算三角形和矩形周長(zhǎng)及面積的方法,涉及java面向?qū)ο笾蓄?、接口、多態(tài)等的使用技巧,需要的朋友可以參考下
    2015-05-05
  • Java數(shù)據(jù)類型Integer與int的區(qū)別詳細(xì)解析

    Java數(shù)據(jù)類型Integer與int的區(qū)別詳細(xì)解析

    這篇文章主要介紹了Java數(shù)據(jù)類型Integer與int的區(qū)別詳細(xì)解析,Ingeter是int的包裝類,int的初值為0,Ingeter的初值為null,int和integer(無(wú)論new否)比,都為true,因?yàn)闀?huì)把Integer自動(dòng)拆箱為int再去比,需要的朋友可以參考下
    2023-12-12
  • IDEA快速顯示Run DashBoard的圖文詳解

    IDEA快速顯示Run DashBoard的圖文詳解

    這篇文章主要介紹了IDEA快速顯示Run DashBoard的圖文詳解,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-12-12
  • 詳解RSA加密算法的原理與Java實(shí)現(xiàn)

    詳解RSA加密算法的原理與Java實(shí)現(xiàn)

    這篇文章主要和大家分享非對(duì)稱加密中的一種算法,那就是 RSA 加密算法。本文介紹了RSA算法的原理與Java實(shí)現(xiàn),感興趣的小伙伴可以嘗試一下
    2022-10-10
  • Java CAS原子操作詳解

    Java CAS原子操作詳解

    在synchronized的優(yōu)化過(guò)程中我們看到大量使用了CAS操作,CAS全稱Compare And Set(或Compare And Swap),簡(jiǎn)單來(lái)說(shuō)CAS操作就是一個(gè)虛擬機(jī)實(shí)現(xiàn)的原子操作
    2023-02-02
  • springboot項(xiàng)目編譯提示無(wú)效的源發(fā)行版17解決辦法

    springboot項(xiàng)目編譯提示無(wú)效的源發(fā)行版17解決辦法

    這篇文章主要給大家介紹了關(guān)于springboot項(xiàng)目編譯提示無(wú)效的源發(fā)行版17解決辦法,這個(gè)錯(cuò)誤意味著你的Spring Boot項(xiàng)目正在使用Java 17這個(gè)版本,但是你的項(xiàng)目中未配置正確的Java版本,需要的朋友可以參考下
    2023-06-06
  • 解決java.lang.NoClassDefFoundError錯(cuò)誤的問(wèn)題

    解決java.lang.NoClassDefFoundError錯(cuò)誤的問(wèn)題

    在Java開(kāi)發(fā)過(guò)程中,NoClassDefFoundError是一個(gè)常見(jiàn)的運(yùn)行時(shí)錯(cuò)誤,是由于JVM在運(yùn)行時(shí)找不到已編譯的類文件導(dǎo)致的,本文就來(lái)介紹一下如何解決,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-09-09
  • Spring之spring-context-indexer依賴詳解

    Spring之spring-context-indexer依賴詳解

    這篇文章主要介紹了Spring之spring-context-indexer依賴詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • Java synchronized同步方法詳解

    Java synchronized同步方法詳解

    這篇文章主要為大家詳細(xì)介紹了Java synchronized同步方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-03-03
  • 關(guān)于Java的HashMap多線程并發(fā)問(wèn)題分析

    關(guān)于Java的HashMap多線程并發(fā)問(wèn)題分析

    HashMap是采用鏈表解決Hash沖突,因?yàn)槭擎湵斫Y(jié)構(gòu),那么就很容易形成閉合的鏈路,這樣在循環(huán)的時(shí)候只要有線程對(duì)這個(gè)HashMap進(jìn)行g(shù)et操作就會(huì)產(chǎn)生死循環(huán),本文針對(duì)這個(gè)問(wèn)題進(jìn)行分析,需要的朋友可以參考下
    2023-05-05

最新評(píng)論