實(shí)例講解Java 自旋鎖
一直以來(lái)不是怎么清楚自旋鎖,最近有點(diǎn)時(shí)間,好好的學(xué)習(xí)了一下;
所謂的自旋鎖在我的理解就是多個(gè)線程在嘗試獲取鎖的時(shí)候,其中一個(gè)線程獲取鎖之后,其他的線程都處在一直嘗試獲取鎖的狀態(tài),不會(huì)阻塞?。。∧敲词裁唇凶鲆恢眹L試獲取鎖呢?就是一個(gè)循環(huán),比較經(jīng)典的是AtomicInteger中的一個(gè)updateAndGet方法,下圖所示(當(dāng)然也可以直接看unsafe類中的getAndAddInt等類似方法);
我們可以看出在while循環(huán)中使用CAS去嘗試更新一個(gè)變量,如果更新失敗,就會(huì)一直在這個(gè)循環(huán)中一直在嘗試;成功的話,就可以到最后的return語(yǔ)句;
由此我們可以大概知道如果自旋的線程過(guò)多,那么CPU的資源就會(huì)被大量消耗!??!

順便提一個(gè)東西叫做原子引用,官方提供了AtomicInteger,AtomicBoolean等原子類,那么如果我們自己定義的類也需要有原子性怎么辦呢?所以官方提供了一個(gè)AtomicReference類,可以將我們自己定義的類封裝一下,就成了我們自己的原子類,例如AtomicReference<Student> atomicReference = new AtomicReference<>();,然后我們對(duì)Student的實(shí)例進(jìn)行CAS各種CAS操作;
栗子:
package TestMain;
import lombok.extern.slf4j.Slf4j;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;
@Slf4j
public class TestMain80 {
//一個(gè)Thread類的原子引用
AtomicReference<Thread> atomicReference = new AtomicReference<>();
//加鎖的方法
public void myLock() {
Thread currentThread = Thread.currentThread();
log.info("myLock--Thread:{}", currentThread.getName());
//這個(gè)就是自旋鎖的核心,利用CAS比較當(dāng)前原子引用中是否為null,如果是null,就把當(dāng)前線程A放到里面去,
// 此時(shí)線程B再到這里,那么就會(huì)CAS失敗,一直在while循環(huán)中
while (!atomicReference.compareAndSet(null, currentThread)) {
}
}
//解鎖的方法
public void myUnlock() {
Thread currentThread = Thread.currentThread();
//CAS比較原子引用中是不是線程A,是的話就更新為null,此時(shí)在上面while中一直在自旋的線程B就可以跳出來(lái)了
atomicReference.compareAndSet(currentThread, null);
log.info("myUnlock--Thread:{}", currentThread.getName());
}
public static void main(String[] args) {
TestMain80 testMain80 = new TestMain80();
//線程A,首先加鎖,然后等3秒中,然后釋放鎖
new Thread(() -> {
testMain80.myLock();
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
testMain80.myUnlock();
}, "A").start();
//主線程等1秒,保證A線程先執(zhí)行
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
//線程B,加鎖再釋放鎖
new Thread(() -> {
testMain80.myLock();
testMain80.myUnlock();
}, "B").start();
}
}

上面的就是一個(gè)自旋鎖的栗子,執(zhí)行結(jié)果中首先是執(zhí)行A線程的myLock方法,獲取鎖成功,之后的B線程雖然也會(huì)執(zhí)行mylock方法,但是會(huì)在while循環(huán)中一直阻塞,直到線程A調(diào)用了myUnlock方法釋放鎖,最后兩行才會(huì)打印出來(lái);
以上就是實(shí)例講解Java 自旋鎖的詳細(xì)內(nèi)容,更多關(guān)于Java 自旋鎖的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
@NonNull導(dǎo)致無(wú)法序列化的問(wèn)題及解決
這篇文章主要介紹了@NonNull導(dǎo)致無(wú)法序列化的問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-01-01
MyBatis中基于別名typeAliases的設(shè)置
這篇文章主要介紹了MyBatis中基于別名typeAliases的設(shè)置,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
Java讀取網(wǎng)絡(luò)文件的實(shí)例代碼
這篇文章主要介紹了Java讀取網(wǎng)絡(luò)文件的實(shí)例代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07
SpringBoot開發(fā)項(xiàng)目,引入JPA找不到findOne方法的解決
這篇文章主要介紹了SpringBoot開發(fā)項(xiàng)目,引入JPA找不到findOne方法的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
Spring Boot 自定義數(shù)據(jù)源DruidDataSource代碼
這篇文章主要介紹了Spring Boot 自定義數(shù)據(jù)源DruidDataSource代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
Spring下Filter過(guò)濾器配置全局異常處理的詳細(xì)步驟
這篇文章主要介紹了Spring下Filter過(guò)濾器配置全局異常處理的詳細(xì)步驟,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12
使用@Autowired可以注入ApplicationContext
這篇文章主要介紹了使用@Autowired可以注入ApplicationContext問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06

