Java實(shí)現(xiàn)手寫自旋鎖的示例代碼
前言
我們?cè)趯懖l(fā)程序的時(shí)候,一個(gè)非常常見的需求就是保證在某一個(gè)時(shí)刻只有一個(gè)線程執(zhí)行某段代碼,像這種代碼叫做臨界區(qū),而通常保證一個(gè)時(shí)刻只有一個(gè)線程執(zhí)行臨界區(qū)的代碼的方法就是鎖。在本篇文章當(dāng)中我們將會(huì)仔細(xì)分析和學(xué)習(xí)自旋鎖,所謂自旋鎖就是通過while循環(huán)實(shí)現(xiàn)的,讓拿到鎖的線程進(jìn)入臨界區(qū)執(zhí)行代碼,讓沒有拿到鎖的線程一直進(jìn)行while死循環(huán),這其實(shí)就是線程自己“旋”在while循環(huán)了,因而這種鎖就叫做自旋鎖。
自旋鎖
原子性
在談自旋鎖之前就不得不談原子性了。所謂原子性簡(jiǎn)單說(shuō)來(lái)就是一個(gè)一個(gè)操作要么不做要么全做,全做的意思就是在操作的過程當(dāng)中不能夠被中斷,比如說(shuō)對(duì)變量data進(jìn)行加一操作,有以下三個(gè)步驟:
- 將data從內(nèi)存加載到寄存器。
- 將data這個(gè)值加一。
- 將得到的結(jié)果寫回內(nèi)存。
原子性就表示一個(gè)線程在進(jìn)行加一操作的時(shí)候,不能夠被其他線程中斷,只有這個(gè)線程執(zhí)行完這三個(gè)過程的時(shí)候其他線程才能夠操作數(shù)據(jù)data。
我們現(xiàn)在用代碼體驗(yàn)一下,在Java當(dāng)中我們可以使用AtomicInteger進(jìn)行對(duì)整型數(shù)據(jù)的原子操作:
import java.util.concurrent.atomic.AtomicInteger; public class AtomicDemo { public static void main(String[] args) throws InterruptedException { AtomicInteger data = new AtomicInteger(); data.set(0); // 將數(shù)據(jù)初始化位0 Thread t1 = new Thread(() -> { for (int i = 0; i < 100000; i++) { data.addAndGet(1); // 對(duì)數(shù)據(jù) data 進(jìn)行原子加1操作 } }); Thread t2 = new Thread(() -> { for (int i = 0; i < 100000; i++) { data.addAndGet(1);// 對(duì)數(shù)據(jù) data 進(jìn)行原子加1操作 } }); // 啟動(dòng)兩個(gè)線程 t1.start(); t2.start(); // 等待兩個(gè)線程執(zhí)行完成 t1.join(); t2.join(); // 打印最終的結(jié)果 System.out.println(data); // 200000 } }
從上面的代碼分析可以知道,如果是一般的整型變量如果兩個(gè)線程同時(shí)進(jìn)行操作的時(shí)候,最終的結(jié)果是會(huì)小于200000。
我們現(xiàn)在來(lái)模擬一下一般的整型變量出現(xiàn)問題的過程:
主內(nèi)存data的初始值等于0,兩個(gè)線程得到的data初始值都等于0。
現(xiàn)在線程一將data加一,然后線程一將data的值同步回主內(nèi)存,整個(gè)內(nèi)存的數(shù)據(jù)變化如下:
現(xiàn)在線程二data加一,然后將data的值同步回主內(nèi)存(將原來(lái)主內(nèi)存的值覆蓋掉了):
我們本來(lái)希望data的值在經(jīng)過上面的變化之后變成2,但是線程二覆蓋了我們的值,因此在多線程情況下,會(huì)使得我們最終的結(jié)果變小。
但是在上面的程序當(dāng)中我們最終的輸出結(jié)果是等于20000的,這是因?yàn)榻odata進(jìn)行+1的操作是原子的不可分的,在操作的過程當(dāng)中其他線程是不能對(duì)data進(jìn)行操作的。這就是原子性帶來(lái)的優(yōu)勢(shì)。
自己動(dòng)手寫自旋鎖
AtomicInteger類
現(xiàn)在我們已經(jīng)了解了原子性的作用了,我們現(xiàn)在來(lái)了解AtomicInteger類的另外一個(gè)原子性的操作——compareAndSet,這個(gè)操作叫做比較并交換(CAS),他具有原子性。
public static void main(String[] args) { AtomicInteger atomicInteger = new AtomicInteger(); atomicInteger.set(0); atomicInteger.compareAndSet(0, 1); }
compareAndSet函數(shù)的意義:首先會(huì)比較第一個(gè)參數(shù)(對(duì)應(yīng)上面的代碼就是0)和atomicInteger的值,如果相等則進(jìn)行交換,也就是將atomicInteger的值設(shè)置為第二個(gè)參數(shù)(對(duì)應(yīng)上面的代碼就是1),如果這些操作成功,那么compareAndSet函數(shù)就返回true,如果操作失敗則返回false,操作失敗可能是因?yàn)榈谝粋€(gè)參數(shù)的值(期望值)和atomicInteger不相等,如果相等也可能因?yàn)樵诟腶tomicInteger的值的時(shí)候失敗(因?yàn)榭赡苡卸鄠€(gè)線程在操作,因?yàn)樵有缘拇嬖?,只能有一個(gè)線程操作成功)。
自旋鎖實(shí)現(xiàn)原理
我們可以使用AtomicInteger類實(shí)現(xiàn)自旋鎖,我們可以用0這個(gè)值表示未上鎖,1這個(gè)值表示已經(jīng)上鎖了。
AtomicInteger類的初始值為0。
在上鎖時(shí),我們可以使用代碼atomicInteger.compareAndSet(0, 1)進(jìn)行實(shí)現(xiàn),我們?cè)谇懊嬉呀?jīng)提到了只能夠有一個(gè)線程完成這個(gè)操作,也就是說(shuō)只能有一個(gè)線程調(diào)用這行代碼然后返回true其余線程都返回false,這些返回false的線程不能夠進(jìn)入臨界區(qū),因此我們需要這些線程停在atomicInteger.compareAndSet(0, 1)這行代碼不能夠往下執(zhí)行,我們可以使用while循環(huán)讓這些線程一直停在這里while (!value.compareAndSet(0, 1));,只有返回true的線程才能夠跳出循環(huán),其余線程都會(huì)一直在這里循環(huán),我們稱這種行為叫做自旋,這種鎖因而也被叫做自旋鎖。
線程在出臨界區(qū)的時(shí)候需要重新將鎖的狀態(tài)調(diào)整為未上鎖的上狀態(tài),我們使用代碼value.compareAndSet(1, 0);就可以實(shí)現(xiàn),將鎖的狀態(tài)還原為未上鎖的狀態(tài),這樣其他的自旋的線程就可以拿到鎖,然后進(jìn)入臨界區(qū)了。
自旋鎖代碼實(shí)現(xiàn)
import java.util.concurrent.atomic.AtomicInteger; public class SpinLock { // 0 表示未上鎖狀態(tài) // 1 表示上鎖狀態(tài) protected AtomicInteger value; public SpinLock() { this.value = new AtomicInteger(); // 設(shè)置 value 的初始值為0 表示未上鎖的狀態(tài) this.value.set(0); } public void lock() { // 進(jìn)行自旋操作 while (!value.compareAndSet(0, 1)); } public void unlock() { // 將鎖的狀態(tài)設(shè)置為未上鎖狀態(tài) value.compareAndSet(1, 0); } }
上面就是我們自己實(shí)現(xiàn)的自旋鎖的代碼,這看起來(lái)實(shí)在太簡(jiǎn)單了,但是它確實(shí)幫助我們實(shí)現(xiàn)了一個(gè)鎖,而且能夠在真實(shí)場(chǎng)景進(jìn)行使用的,我們現(xiàn)在用代碼對(duì)上面我們寫的鎖進(jìn)行測(cè)試。
測(cè)試程序:
public class SpinLockTest { public static int data; public static SpinLock lock = new SpinLock(); public static void add() { for (int i = 0; i < 100000; i++) { // 上鎖 只能有一個(gè)線程執(zhí)行 data++ 操作 其余線程都只能進(jìn)行while循環(huán) lock.lock(); data++; lock.unlock(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[100]; // 設(shè)置100個(gè)線程 for (int i = 0; i < 100; i ++) { threads[i] = new Thread(SpinLockTest::add); } // 啟動(dòng)一百個(gè)線程 for (int i = 0; i < 100; i++) { threads[i].start(); } // 等待這100個(gè)線程執(zhí)行完成 for (int i = 0; i < 100; i++) { threads[i].join(); } System.out.println(data); // 10000000 } }
在上面的代碼單中,我們使用100個(gè)線程,然后每個(gè)線程循環(huán)執(zhí)行100000data++操作,上面的代碼最后輸出的結(jié)果是10000000,和我們期待的結(jié)果是相等的,這就說(shuō)明我們實(shí)現(xiàn)的自旋鎖是正確的。
自己動(dòng)手寫可重入自旋鎖
可重入自旋鎖
在上面實(shí)現(xiàn)的自旋鎖當(dāng)中已經(jīng)可以滿足一些我們的基本需求了,就是一個(gè)時(shí)刻只能夠有一個(gè)線程執(zhí)行臨界區(qū)的代碼。但是上面的的代碼并不能夠滿足重入的需求,也就是說(shuō)上面寫的自旋鎖并不是一個(gè)可重入的自旋鎖,事實(shí)上在上面實(shí)現(xiàn)的自旋鎖當(dāng)中重入的話就會(huì)產(chǎn)生死鎖。
我們通過一份代碼來(lái)模擬上面重入產(chǎn)生死鎖的情況:
public static void add(int state) throws InterruptedException { TimeUnit.SECONDS.sleep(1); if (state <= 3) { lock.lock(); System.out.println(Thread.currentThread().getName() + "\t進(jìn)入臨界區(qū) state = " + state); for (int i = 0; i < 10; i++) data++; add(state + 1); // 進(jìn)行遞歸重入 重入之前鎖狀態(tài)已經(jīng)是1了 因?yàn)檫@個(gè)線程進(jìn)入了臨界區(qū) lock.unlock(); } }
在上面的代碼當(dāng)中加入我們傳入的參數(shù)state的值為1,那么在線程執(zhí)行for循環(huán)之后再次遞歸調(diào)用add函數(shù)的話,那么state的值就變成了2。
if條件仍然滿足,這個(gè)線程也需要重新獲得鎖,但是此時(shí)鎖的狀態(tài)是1,這個(gè)線程已經(jīng)獲得過一次鎖了,但是自旋鎖期待的鎖的狀態(tài)是0,因?yàn)橹挥羞@樣他才能夠再次獲得鎖,進(jìn)入臨界區(qū),但是現(xiàn)在鎖的狀態(tài)是1,也就是說(shuō)雖然這個(gè)線程獲得過一次鎖,但是它也會(huì)一直進(jìn)行while循環(huán)而且永遠(yuǎn)都出不來(lái)了,這樣就形成了死鎖了。
可重入自旋鎖思想
針對(duì)上面這種情況我們需要實(shí)現(xiàn)一個(gè)可重入的自旋鎖,我們的思想大致如下:
- 在我們實(shí)現(xiàn)的自旋鎖當(dāng)中,我們可以增加兩個(gè)變量,owner一個(gè)用于存當(dāng)前擁有鎖的線程,count一個(gè)記錄當(dāng)前線程進(jìn)入鎖的次數(shù)。
- 如果線程獲得鎖,owner = Thread.currentThread()并且count = 1。
- 當(dāng)線程下次再想獲取鎖的時(shí)候,首先先看owner是不是指向自己,則一直進(jìn)行循環(huán)操作,如果是則直接進(jìn)行count++操作,然后就可以進(jìn)入臨界區(qū)了。
- 我們?cè)诔雠R界區(qū)的時(shí)候,如果count大于一的話,說(shuō)明這個(gè)線程重入了這把鎖,因此不能夠直接將鎖設(shè)置為0也就是未上鎖的狀態(tài),這種情況直接進(jìn)行count--操作,如果count等于1的話,說(shuō)明線程當(dāng)前的狀態(tài)不是重入狀態(tài)(可能是重入之后遞歸返回了),因此在出臨界區(qū)之前需要將鎖的狀態(tài)設(shè)置為0,也就是沒上鎖的狀態(tài),好讓其他線程能夠獲取鎖。
可重入鎖代碼實(shí)現(xiàn)
實(shí)現(xiàn)的可重入鎖代碼如下:
public class ReentrantSpinLock extends SpinLock { private Thread owner; private int count; @Override public void lock() { if (owner == null || owner != Thread.currentThread()) { while (!value.compareAndSet(0, 1)); owner = Thread.currentThread(); count = 1; }else { count++; } } @Override public void unlock() { if (count == 1) { count = 0; value.compareAndSet(1, 0); }else count--; } }
下面我們通過一個(gè)遞歸程序去驗(yàn)證我們寫的可重入的自旋鎖是否能夠成功工作。
測(cè)試程序:
import java.util.concurrent.TimeUnit; public class ReentrantSpinLockTest { public static int data; public static ReentrantSpinLock lock = new ReentrantSpinLock(); public static void add(int state) throws InterruptedException { TimeUnit.SECONDS.sleep(1); if (state <= 3) { lock.lock(); System.out.println(Thread.currentThread().getName() + "\t進(jìn)入臨界區(qū) state = " + state); for (int i = 0; i < 10; i++) data++; add(state + 1); lock.unlock(); } } public static void main(String[] args) throws InterruptedException { Thread[] threads = new Thread[10]; for (int i = 0; i < 10; i++) { threads[i] = new Thread(new Thread(() -> { try { ReentrantSpinLockTest.add(1); } catch (InterruptedException e) { e.printStackTrace(); } }, String.valueOf(i))); } for (int i = 0; i < 10; i++) { threads[i].start(); } for (int i = 0; i < 10; i++) { threads[i].join(); } System.out.println(data); } }
上面程序的輸出:
Thread-3 進(jìn)入臨界區(qū) state = 1
Thread-3 進(jìn)入臨界區(qū) state = 2
Thread-3 進(jìn)入臨界區(qū) state = 3
Thread-0 進(jìn)入臨界區(qū) state = 1
Thread-0 進(jìn)入臨界區(qū) state = 2
Thread-0 進(jìn)入臨界區(qū) state = 3
Thread-9 進(jìn)入臨界區(qū) state = 1
Thread-9 進(jìn)入臨界區(qū) state = 2
Thread-9 進(jìn)入臨界區(qū) state = 3
Thread-4 進(jìn)入臨界區(qū) state = 1
Thread-4 進(jìn)入臨界區(qū) state = 2
Thread-4 進(jìn)入臨界區(qū) state = 3
Thread-7 進(jìn)入臨界區(qū) state = 1
Thread-7 進(jìn)入臨界區(qū) state = 2
Thread-7 進(jìn)入臨界區(qū) state = 3
Thread-8 進(jìn)入臨界區(qū) state = 1
Thread-8 進(jìn)入臨界區(qū) state = 2
Thread-8 進(jìn)入臨界區(qū) state = 3
Thread-5 進(jìn)入臨界區(qū) state = 1
Thread-5 進(jìn)入臨界區(qū) state = 2
Thread-5 進(jìn)入臨界區(qū) state = 3
Thread-2 進(jìn)入臨界區(qū) state = 1
Thread-2 進(jìn)入臨界區(qū) state = 2
Thread-2 進(jìn)入臨界區(qū) state = 3
Thread-6 進(jìn)入臨界區(qū) state = 1
Thread-6 進(jìn)入臨界區(qū) state = 2
Thread-6 進(jìn)入臨界區(qū) state = 3
Thread-1 進(jìn)入臨界區(qū) state = 1
Thread-1 進(jìn)入臨界區(qū) state = 2
Thread-1 進(jìn)入臨界區(qū) state = 3
300
從上面的輸出結(jié)果我們就可以知道,當(dāng)一個(gè)線程能夠獲取鎖的時(shí)候他能夠進(jìn)行重入,而且最終輸出的結(jié)果也是正確的,因此驗(yàn)證了我們寫了可重入自旋鎖是有效的!
總結(jié)
在本篇文章當(dāng)中主要給大家介紹了自旋鎖和可重入自旋鎖的原理,并且實(shí)現(xiàn)了一遍,其實(shí)代碼還是比較簡(jiǎn)單關(guān)鍵需要大家將這其中的邏輯理清楚:
所謂自旋鎖就是通過while循環(huán)實(shí)現(xiàn)的,讓拿到鎖的線程進(jìn)入臨界區(qū)執(zhí)行代碼,讓沒有拿到鎖的線程一直進(jìn)行while死循環(huán)。
可重入的含義就是一個(gè)線程已經(jīng)競(jìng)爭(zhēng)到了一個(gè)鎖,在競(jìng)爭(zhēng)到這個(gè)鎖之后又一次有重入臨界區(qū)代碼的需求,如果能夠保證這個(gè)線程能夠重新進(jìn)入臨界區(qū),這就叫可重入。
我們?cè)趯?shí)現(xiàn)自旋鎖的時(shí)候使用的是AtomicInteger類,并且我們使用0和1這兩個(gè)數(shù)值用于表示無(wú)鎖和鎖被占用兩個(gè)狀態(tài),在獲取鎖的時(shí)候使用while循環(huán)不斷進(jìn)行CAS操作,直到操作成功返回true,在釋放鎖的時(shí)候使用CAS將鎖的狀態(tài)從1變成0。
實(shí)現(xiàn)可重入鎖最重要的一點(diǎn)就是需要記錄是那個(gè)線程獲得了鎖,同時(shí)還需要記錄獲取了幾次鎖,因?yàn)槲覀冊(cè)诮怄i的時(shí)候需要進(jìn)行判斷,之后count = 1的情況才能將鎖的狀態(tài)從1設(shè)置成0。
到此這篇關(guān)于Java實(shí)現(xiàn)手寫自旋鎖的示例代碼的文章就介紹到這了,更多相關(guān)Java自旋鎖內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java abstract class 與 interface對(duì)比
這篇文章主要介紹了 Java abstract class 與 interface對(duì)比的相關(guān)資料,需要的朋友可以參考下2016-12-12基于JavaScript動(dòng)態(tài)規(guī)劃編寫一個(gè)益智小游戲
最近在學(xué)習(xí)動(dòng)態(tài)規(guī)劃相關(guān)的知識(shí),所以本文將利用動(dòng)態(tài)規(guī)劃編寫一個(gè)簡(jiǎn)單的益智小游戲,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-06-06mybatis typeAliases 給實(shí)體類起別名的方法
這篇文章主要介紹了mybatis typeAliases 給實(shí)體類起別名,本文給大家分享兩種用法,通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09spring boot中xalan引入報(bào)錯(cuò)系統(tǒng)找不到指定的文件原因分析
這篇文章主要介紹了spring boot中xalan引入報(bào)錯(cuò)系統(tǒng)找不到指定的文件,主要原因是內(nèi)嵌的tomcat9.0.36,本文給大家分享最新解決方法,需要的朋友可以參考下2023-08-08Java利用Poi讀取excel并對(duì)所有類型進(jìn)行處理
這篇文章主要為大家詳細(xì)介紹了Java利用Poi讀取excel并對(duì)所有類型進(jìn)行處理的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2024-01-01Java模擬實(shí)現(xiàn)QQ三方登錄(單點(diǎn)登錄2.0)
這篇文章主要為大家詳細(xì)介紹了Java模擬實(shí)現(xiàn)QQ三方登錄,單點(diǎn)登錄2.0,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-06-06java 垃圾回收機(jī)制以及經(jīng)典垃圾回收器詳解
這篇文章主要介紹了java 垃圾回收機(jī)制以及經(jīng)典垃圾回收器詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07java中springMVC獲取請(qǐng)求參數(shù)的方法
這篇文章主要介紹了java中springMVC獲取請(qǐng)求參數(shù)的方法,springmvc是spring框架的一個(gè)模塊,springmvc和spring無(wú)需通過中間整合層進(jìn)行整合,需要的朋友可以參考下2023-05-05