實(shí)例講解Java 自旋鎖
一直以來不是怎么清楚自旋鎖,最近有點(diǎn)時間,好好的學(xué)習(xí)了一下;
所謂的自旋鎖在我的理解就是多個線程在嘗試獲取鎖的時候,其中一個線程獲取鎖之后,其他的線程都處在一直嘗試獲取鎖的狀態(tài),不會阻塞!??!那么什么叫做一直嘗試獲取鎖呢?就是一個循環(huán),比較經(jīng)典的是AtomicInteger中的一個updateAndGet方法,下圖所示(當(dāng)然也可以直接看unsafe類中的getAndAddInt等類似方法);
我們可以看出在while循環(huán)中使用CAS去嘗試更新一個變量,如果更新失敗,就會一直在這個循環(huán)中一直在嘗試;成功的話,就可以到最后的return語句;
由此我們可以大概知道如果自旋的線程過多,那么CPU的資源就會被大量消耗!?。?/p>
順便提一個東西叫做原子引用,官方提供了AtomicInteger,AtomicBoolean等原子類,那么如果我們自己定義的類也需要有原子性怎么辦呢?所以官方提供了一個AtomicReference類,可以將我們自己定義的類封裝一下,就成了我們自己的原子類,例如AtomicReference<Student> atomicReference = new AtomicReference<>();
,然后我們對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 { //一個Thread類的原子引用 AtomicReference<Thread> atomicReference = new AtomicReference<>(); //加鎖的方法 public void myLock() { Thread currentThread = Thread.currentThread(); log.info("myLock--Thread:{}", currentThread.getName()); //這個就是自旋鎖的核心,利用CAS比較當(dāng)前原子引用中是否為null,如果是null,就把當(dāng)前線程A放到里面去, // 此時線程B再到這里,那么就會CAS失敗,一直在while循環(huán)中 while (!atomicReference.compareAndSet(null, currentThread)) { } } //解鎖的方法 public void myUnlock() { Thread currentThread = Thread.currentThread(); //CAS比較原子引用中是不是線程A,是的話就更新為null,此時在上面while中一直在自旋的線程B就可以跳出來了 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(); } }
上面的就是一個自旋鎖的栗子,執(zhí)行結(jié)果中首先是執(zhí)行A線程的myLock方法,獲取鎖成功,之后的B線程雖然也會執(zhí)行mylock方法,但是會在while循環(huán)中一直阻塞,直到線程A調(diào)用了myUnlock方法釋放鎖,最后兩行才會打印出來;
以上就是實(shí)例講解Java 自旋鎖的詳細(xì)內(nèi)容,更多關(guān)于Java 自旋鎖的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
springboot整合JPA訪問Mysql的實(shí)現(xiàn)方法
本文主要介紹了springboot整合JPA訪問Mysql的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-02-02使用Java的Spring框架編寫第一個程序Hellow world
這篇文章主要介紹了Java的Spring框架并用其開始編寫第一個程序Hellow world的方法,Spring是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下2015-12-12Java如何解決發(fā)送Post請求報(bào)Stream?closed問題
這篇文章主要介紹了Java如何解決發(fā)送Post請求報(bào)Stream?closed問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06Java SpringBoot實(shí)現(xiàn)AOP
AOP包括連接點(diǎn)(JoinPoint)、切入點(diǎn)(Pointcut)、增強(qiáng)(Advisor)、切面(Aspect)、AOP代理(AOP Proxy),具體的方法和類型下面文章會舉例說明,感興趣的小伙伴和小編一起閱讀全文吧2021-09-09java中orElse和orElseGet方法區(qū)別小結(jié)
這篇文章主要給大家介紹了關(guān)于java中orElse和orElseGet方法區(qū)別的相關(guān)資料,兩者之間的區(qū)別細(xì)微,但是卻在某些場景下顯的很重要,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-10-10