摘要:公平鎖阻塞隊(duì)列前邊有線程,要去后邊排隊(duì),簡單來說滾后邊等著去。非公平鎖不管是否有線程排隊(duì),先槍鎖基于實(shí)現(xiàn)的可重入鎖實(shí)現(xiàn)類。
AQS (AbstractQueuedSynchronizer)
底層一個(gè)隊(duì)列 阻塞隊(duì)列 ->
? Abstract:因?yàn)樗⒉恢涝趺瓷湘i。模板方法設(shè)計(jì)模式即可,暴露出鎖邏輯。
? Queue :線程阻塞隊(duì)列 Synchronizer:同步
? CAS + state 完成多線程槍鎖邏輯 Queue 完成搶不到鎖的線程排隊(duì)
//獲取鎖public final void acquire(int arg) { if (!tryAcquire(arg) && // 子類判定獲取鎖是否失敗 acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) // 獲取失敗后添加到阻塞隊(duì)列 selfInterrupt();}// 子類實(shí)現(xiàn)獲取鎖的邏輯,AQS并不知道你怎么用這個(gè)state來上鎖protected boolean tryAcquire(int arg) { throw new UnsupportedOperationException();}//釋放鎖的public final boolean release(int arg) { // 子類判定釋放鎖成功 if (tryRelease(arg)) { // 檢查阻塞隊(duì)列喚醒即可 Node h = head; if (h != null && h.waitStatus != 0) unparkSuccessor(h); return true; } return false;}// 子類實(shí)現(xiàn)獲取鎖的邏輯protected boolean tryRelease(int arg) { throw new UnsupportedOperationException();}
子類只需要實(shí)現(xiàn)自己的獲取鎖邏輯和釋放鎖邏輯即可,至于排隊(duì)阻塞等待、喚醒機(jī)制均由AQS來完成。
公平鎖:阻塞隊(duì)列前邊有線程,要去后邊排隊(duì),簡單來說 滾后邊等著去。(FIFO) 非公平鎖:不管是否有線程排隊(duì),先槍鎖
基于AQS實(shí)現(xiàn)的可重入鎖實(shí)現(xiàn)類。
ReentrantLock 是一個(gè)可重入的互斥(/獨(dú)占)鎖,又稱為“獨(dú)占鎖”
重入鎖: 自己可以再次獲取自己的內(nèi)部的鎖。
state 來表示鎖狀態(tài)和重入次數(shù),0無鎖,大于0 重入次數(shù),1為重入1次,也即只加鎖一次
private final Sync sync;//sync,構(gòu)造方法中初始化,決定使用公平鎖還是非公平鎖的方式獲取鎖。
public ReentrantLock() { // 默認(rèn)為非公平鎖。 sync = new NonfairSync(); } public ReentrantLock(boolean fair) { // 由fair變量來表明選擇鎖類型 sync = fair ? new FairSync() : new NonfairSync(); }
abstract static class Sync extends AbstractQueuedSynchronizer {}
static final class NonfairSync extends Sync {} 非公平鎖
static final class FairSync extends Sync {} 公平鎖
//非公平鎖 static final class NonfairSync extends Sync { // 由ReentrantLock調(diào)用獲取鎖 final void lock() { // 非公平鎖,直接搶鎖,不管有沒有線程排隊(duì) if (compareAndSetState(0, 1)) // 上鎖成功,標(biāo)識(shí)當(dāng)前線程為獲取鎖的線程 setExclusiveOwnerThread(Thread.currentThread()); else // 搶鎖失敗,進(jìn)入AQS的標(biāo)準(zhǔn)獲取鎖流程 acquire(1); } protected final boolean tryAcquire(int acquires) { // 使用父類提供的獲取非公平鎖的方法來獲取鎖 return nonfairTryAcquire(acquires); } } //公平鎖 static final class FairSync extends Sync { // 由ReentrantLock調(diào)用 final void lock() { // 沒有嘗試搶鎖,直接進(jìn)入AQS標(biāo)準(zhǔn)獲取鎖流程 acquire(1); } // AQS調(diào)用,子類自己實(shí)現(xiàn)獲取鎖的流程 protected final boolean tryAcquire(int acquires) { final Thread current = Thread.currentThread(); int c = getState(); // 此時(shí)有可能正好獲取鎖的線程釋放了鎖,也有可能本身就沒有線程獲取鎖 if (c == 0) { //這里和非公平鎖的區(qū)別在于:hasQueuedPredecessors看看隊(duì)列中是否有線程正在排隊(duì),沒有的話再通過CAS搶鎖 if (!hasQueuedPredecessors() && compareAndSetState(0, acquires)) { setExclusiveOwnerThread(current); return true; } } // 當(dāng)前線程就是獲取鎖的線程,那么這里是鎖重入,和非公平鎖操作一模一樣 else if (current == getExclusiveOwnerThread()) { int nextc = c + acquires; if (nextc < 0) throw new Error("Maximum lock count exceeded"); setState(nextc); return true; } //需要AQS來將當(dāng)前線程放入阻塞隊(duì)列,然后進(jìn)行阻塞操作等待喚醒獲取鎖 return false; } } abstract static class Sync extends AbstractQueuedSynchronizer { abstract void lock(); // 非公平鎖標(biāo)準(zhǔn)獲取鎖方法 final boolean nonfairTryAcquire(int acquires) { final Thread current = Thread.currentThread(); int c = getState(); //獲取所得線程釋放了鎖,那么可以嘗試搶鎖 if (c == 0) { // 繼續(xù)搶鎖,不看有沒有線程排隊(duì) if (compareAndSetState(0, acquires)) { setExclusiveOwnerThread(current); return true; } } // 當(dāng)前線程就是持有鎖的線程,表明鎖重入 else if (current == getExclusiveOwnerThread()) { // 利用state 進(jìn)行次數(shù)記錄 int nextc = c + acquires; // 如果超過了int表示范圍,表明符號(hào)溢出,所以拋出異常 if (nextc < 0) throw new Error("Maximum lock count exceeded"); setState(nextc);//state 變量賦值 return true; } //表明需要AQS來將當(dāng)前線程放入阻塞隊(duì)列,然后進(jìn)行阻塞操作等待喚醒獲取鎖 return false; } // 公平鎖和非公平鎖公用方法,在釋放鎖的時(shí)候,并不區(qū)分是否公平 protected final boolean tryRelease(int releases) { int c = getState() - releases; // 如果當(dāng)前線程不是上鎖的那個(gè)線程 if (Thread.currentThread() != getExclusiveOwnerThread()) throw new IllegalMonitorStateException(); boolean free = false; // 不是重入鎖,那么當(dāng)前線程一定是釋放鎖了,把當(dāng)前AQS用于保存當(dāng)前鎖對(duì)象的變量ExclusiveOwnerThread設(shè)置為null,表明釋放鎖成功 if (c == 0) { free = true; setExclusiveOwnerThread(null); } //此時(shí)state全局變量沒有改變,也就意味著在setState之前 //沒有別的線程能夠獲取鎖,保證了以上的操作原子性 setState(c); //釋放鎖成功了,可以去喚醒正在等待鎖的線程 return free; } protected final boolean isHeldExclusively() { return getExclusiveOwnerThread() == Thread.currentThread(); } final ConditionObject newCondition() { return new ConditionObject(); } }
public void lock() {//獲取鎖的操作 // 直接通過sync同步器上鎖 sync.lock();}public void unlock() {//釋放鎖的操作 sync.release(1);}
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/125657.html
摘要:簡介繼續(xù)分析源碼,上一篇文章把的分析完畢。本文開始分析簡單的介紹一下。存儲(chǔ)的元素是無序的并且允許使用空的元素。 1.簡介 繼續(xù)分析源碼,上一篇文章把HashMap的分析完畢。本文開始分析HashSet簡單的介紹一下。 HashSet是一個(gè)無重復(fù)元素集合,內(nèi)部使用HashMap實(shí)現(xiàn),所以HashMap的特征耶繼承了下來。存儲(chǔ)的元素是無序的并且HashSet允許使用空的元素。 HashSe...
摘要:則使用了拉鏈?zhǔn)降纳⒘兴惴ǎ⒃谥幸肓思t黑樹優(yōu)化過長的鏈表。如果大家對(duì)紅黑樹感興趣,可以閱讀我的另一篇文章紅黑樹詳細(xì)分析。構(gòu)造方法構(gòu)造方法分析的構(gòu)造方法不多,只有四個(gè)。 1.概述 本篇文章我們來聊聊大家日常開發(fā)中常用的一個(gè)集合類 - HashMap。HashMap 最早出現(xiàn)在 JDK 1.2中,底層基于散列算法實(shí)現(xiàn)。HashMap 允許 null 鍵和 null 值,在計(jì)算哈鍵的哈希值...
摘要:所謂拉鏈法就是將鏈表和數(shù)組相結(jié)合。若遇到哈希沖突,則將沖突的值加到鏈表中即可。在編寫程序中,要盡量避免。 目錄: 0-1. 簡介 0-2. 內(nèi)部結(jié)構(gòu)分析 0-2-1. JDK18之前 0-2-2. JDK18之后 0-3. LinkedList源碼分析 0-3-1. 構(gòu)造方法 0-3-2. put方法 0-3-3. get方法 0-3-4. resize方法 ...
摘要:唐老師,回答道讀源碼是要建立在你的基礎(chǔ)經(jīng)驗(yàn)足夠的情況下。除了自己去閱讀源碼之外,比如學(xué)習(xí)某個(gè)類的時(shí)候,可以專門結(jié)合一些優(yōu)質(zhì)的博客針對(duì)性的對(duì)比學(xué)習(xí),并查漏補(bǔ)缺。制定源碼學(xué)習(xí)計(jì)劃。多調(diào)試,跟蹤源碼。如若有好的學(xué)習(xí)方法,可以留言一起交流學(xué)習(xí)。 序言:目前看一看源碼,來提升自己的技術(shù)實(shí)力。同時(shí)現(xiàn)在好多面試官都喜歡問源碼,問你是否讀過JDK源碼等等? 針對(duì)如何閱讀源碼,也請(qǐng)教了我的老師。下面就先...
閱讀 3800·2023-01-11 11:02
閱讀 4306·2023-01-11 11:02
閱讀 3128·2023-01-11 11:02
閱讀 5237·2023-01-11 11:02
閱讀 4801·2023-01-11 11:02
閱讀 5574·2023-01-11 11:02
閱讀 5379·2023-01-11 11:02
閱讀 4080·2023-01-11 11:02