摘要:等到所有子線程都執(zhí)行完后即,會主調(diào)用線程,然后主調(diào)用線程就會從函數(shù)返回,繼續(xù)后余動作。
原理剖析(第 005 篇)AQS工作原理分析
-
一、大致介紹1、前面章節(jié)講解了一下CAS,簡單講就是cmpxchg+lock的原子操作; 2、而在談到并發(fā)操作里面,我們不得不談到AQS,JDK的源碼里面好多并發(fā)的類都是通過Sync的內(nèi)部類繼承AQS而實現(xiàn)出五花八門的功能; 3、本章節(jié)就和大家分享分析一下AQS的工作原理;二、簡單認(rèn)識AQS 2.1 何為AQS?
1、AQS是一個抽象類,類名為AbstractQueuedSynchronizer,抽象的都是一些公用的方法屬性,其自身是沒有實現(xiàn)任何同步接口的; 2、AQS定義了同步器中獲取鎖和釋放鎖,目的來讓自定義同步器組件來使用或重寫; 3、縱觀AQS的子類,絕大多數(shù)都是一個叫Sync的靜態(tài)內(nèi)部類來繼承AQS類,通過重寫AQS中的一些方法來實現(xiàn)自定義同步器; 4、AQS定義了兩種資源共享方式:EXCLUSIVE( 獨占式:每次僅有一個Thread能執(zhí)行 )、SHARED( 共享式:多個線程可同時執(zhí)行 ); 5、AQS維護(hù)了一個FIFO的CLH鏈表隊列,且該隊列不支持基于優(yōu)先級的同步策略;2.2 AQS的state關(guān)鍵詞
1、private volatile int state:維護(hù)了一個volatile的int類型的state字段,該字段是實現(xiàn)AQS的核心關(guān)鍵詞; 2、通過getState、setState、compareAndSetState方法類獲取、設(shè)置更新state值; 3、該字段在不同的并發(fā)類中起著不同的紐帶作用,下面會接著講到state字段的一些應(yīng)用場景;2.3 Node的waitStatus關(guān)鍵詞
1、正常默認(rèn)的狀態(tài)值為0; 2、對于釋放操作的時候,前一個結(jié)點有喚醒后一個結(jié)點的任務(wù); 3、當(dāng)前結(jié)點的前置結(jié)點waitStatus > 0,則結(jié)點處于CANCELLED狀態(tài),應(yīng)該需要踢出隊列; 4、當(dāng)前結(jié)點的前置結(jié)點waitStatus = 0,則需要將前置結(jié)點改為SIGNAL狀態(tài);2.4 CLH隊列
1、隊列模型: +------+ prev +------+ prev +------+ | | <---- | | <---- | | head | Node | next | Node | next | Node | tail | | ----> | | ----> | | +------+ +------+ +------+ 2、鏈表結(jié)構(gòu),在頭尾結(jié)點中,需要特別指出的是頭結(jié)點是一個空對象結(jié)點,無任何意義,即傀儡結(jié)點; 3、每一個Node結(jié)點都維護(hù)了一個指向前驅(qū)的指針和指向后驅(qū)的指針,結(jié)點與結(jié)點之間相互關(guān)聯(lián)構(gòu)成鏈表; 4、入隊在尾,出隊在頭,出隊后需要激活該出隊結(jié)點的后繼結(jié)點,若后繼結(jié)點為空或后繼結(jié)點waitStatus>0,則從隊尾向前遍歷取waitStatus<0的觸發(fā)阻塞喚醒;2.5 state在AQS簡單應(yīng)用舉例
1、CountDownLatch,簡單大致意思為:A組線程等待另外B組線程,B組線程執(zhí)行完了,A組線程才可以執(zhí)行; state初始化假設(shè)為N,后續(xù)每countDown()一次,state會CAS減1。 等到所有子線程都執(zhí)行完后(即state=0),會unpark()主調(diào)用線程,然后主調(diào)用線程就會從await()函數(shù)返回,繼續(xù)后余動作。 2、ReentrantLock,簡單大致意思為:獨占式鎖的類; state初始化為0,表示未鎖定狀態(tài),然后每lock()時調(diào)用tryAcquire()使state加1, 其他線程再tryAcquire()時就會失敗,直到A線程unlock()到state=0(即釋放鎖)為止,其它線程才有機(jī)會獲取該鎖; 3、Semaphore,簡單大致意思為:A、B、C、D線程同時爭搶資源,目前卡槽大小為2,若A、B正在執(zhí)行且未執(zhí)行完,那么C、D線程在門外等著,一旦A、B有1個執(zhí)行完了,那么C、D就會競爭看誰先執(zhí)行; state初始值假設(shè)為N,后續(xù)每tryAcquire()一次,state會CAS減1,當(dāng)state為0時其它線程處于等待狀態(tài), 直到state>0且2.6 常用重要的方法 1、protected boolean isHeldExclusively() // 需要被子類實現(xiàn)的方法,調(diào)用該方法的線程是否持有獨占鎖,一般用到了condition的時候才需要實現(xiàn)此方法 2、protected boolean tryAcquire(int arg) // 需要被子類實現(xiàn)的方法,獨占方式嘗試獲取鎖,獲取鎖成功后返回true,獲取鎖失敗后返回false 3、protected boolean tryRelease(int arg) // 需要被子類實現(xiàn)的方法,獨占方式嘗試釋放鎖,釋放鎖成功后返回true,釋放鎖失敗后返回false 4、protected int tryAcquireShared(int arg) // 需要被子類實現(xiàn)的方法,共享方式嘗試獲取鎖,獲取鎖成功后返回正數(shù)1,獲取鎖失敗后返回負(fù)數(shù)-1 5、protected boolean tryReleaseShared(int arg) // 需要被子類實現(xiàn)的方法,共享方式嘗試釋放鎖,釋放鎖成功后返回正數(shù)1,釋放鎖失敗后返回負(fù)數(shù)-1 6、final boolean acquireQueued(final Node node, int arg) // 對于進(jìn)入隊尾的結(jié)點,檢測自己可以休息了,如果可以修改則進(jìn)入SIGNAL狀態(tài)且進(jìn)入park()阻塞狀態(tài) 7、private Node addWaiter(Node mode) // 添加結(jié)點到鏈表隊尾 8、private Node enq(final Node node) // 如果addWaiter嘗試添加隊尾失敗,則再次調(diào)用enq此方法自旋將結(jié)點加入隊尾 9、private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) // 檢測結(jié)點狀態(tài),如果可以休息的話則設(shè)置waitStatus=SIGNAL并調(diào)用LockSupport.park休息; 10、private void unparkSuccessor(Node node) // 釋放鎖時,該方法需要負(fù)責(zé)喚醒后繼節(jié)點2.7 設(shè)計與實現(xiàn)偽代碼1、獲取獨占鎖: public final void acquire(int arg) { if (!tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) selfInterrupt(); } acquire{ 如果嘗試獲取獨占鎖失敗的話( 嘗試獲取獨占鎖的各種方式由AQS的子類實現(xiàn) ), 那么就新增獨占鎖結(jié)點通過自旋操作加入到隊列中,并且根據(jù)結(jié)點中的waitStatus來決定是否調(diào)用LockSupport.park進(jìn)行休息 } 2、釋放獨占鎖: public final boolean release(int arg) { if (tryRelease(arg)) { Node h = head; if (h != null && h.waitStatus != 0) unparkSuccessor(h); return true; } return false; } release{ 如果嘗試釋放獨占鎖成功的話( 嘗試釋放獨占鎖的各種方式由AQS的子類實現(xiàn) ), 那么取出頭結(jié)點并根據(jù)結(jié)點waitStatus來決定是否有義務(wù)喚醒其后繼結(jié)點 } 3、獲取共享鎖: public final void acquireShared(int arg) { if (tryAcquireShared(arg) < 0) doAcquireShared(arg); } acquireShared{ 如果嘗試獲取共享鎖失敗的話( 嘗試獲取共享鎖的各種方式由AQS的子類實現(xiàn) ), 那么新增共享鎖結(jié)點通過自旋操作加入到隊尾中,并且根據(jù)結(jié)點中的waitStatus來決定是否調(diào)用LockSupport.park進(jìn)行休息 } 4、釋放共享鎖: public final boolean releaseShared(int arg) { if (tryReleaseShared(arg)) { doReleaseShared(); return true; } return false; } releaseShared{ 如果嘗試釋放共享鎖失敗的話( 嘗試釋放共享鎖的各種方式由AQS的子類實現(xiàn) ), 那么通過自旋操作喚完成阻塞線程的喚起操作 }三、舉例ReentrantLock 3.1、ReentrantLock1、在分析AQS源碼前,我們需要依賴一個載體來說,畢竟AQS的一些方法都是空方法且拋異常的,所以單講AQS不太生動形象; 2、因此我們決定采用ReentrantLock來講解,其他都大致差不多,因為了解了一個,其他都可以依葫蘆畫瓢秒懂;3.2、ReentrantLock生活細(xì)節(jié)化理解比如我們天天在外面吃快餐,我就以吃快餐為例生活化闡述該ReentrantLock原理: 1、場景:餐廳只有一個排隊的走廊,只有一個打飯菜的師傅; 2、開飯時間點,大家都爭先恐后的去吃飯,因此排上了隊,挨個挨個排隊打飯菜,任何一個人只要排到了打飯師傅的前面,都可以打到飯菜; 3、但是有時候隊很長,有些人之間的關(guān)系是家屬關(guān)系,如果后來的人看到自己家屬正在打飯菜,這個時候可以不用排隊直接跑到前面打飯菜; 4、總之大家都挨個挨個排隊打飯,有家屬關(guān)系的直接跑到前面打飯菜; 5、到此打止,1、2、3、4可以認(rèn)為是一種公平方式的獨占鎖,3可以理解為重入鎖; 5、但是呢,還有那么些緊急趕時間的人,而且又跟排隊的人沒半點瓜葛,來餐廳時剛好看到師傅剛剛打完一個人的飯菜,于是插入去打飯菜敢時間; 6、如果敢時間人的來的時候發(fā)現(xiàn)師傅還在打飯菜,那么就只得乖乖的排隊等候打飯菜咯; 7、到此打止,1、2、5、6可以認(rèn)為是一種非公平方式的獨占鎖;四、源碼分析ReentrantLock 4.1、ReentrantLock構(gòu)造器1、構(gòu)造器源碼: /** * Creates an instance of {@code ReentrantLock}. * This is equivalent to using {@code ReentrantLock(false)}. */ public ReentrantLock() { sync = new NonfairSync(); } /** * Creates an instance of {@code ReentrantLock} with the * given fairness policy. * * @param fair {@code true} if this lock should use a fair ordering policy */ public ReentrantLock(boolean fair) { sync = fair ? new FairSync() : new NonfairSync(); } 2、默認(rèn)構(gòu)造方法為非公平鎖,帶參構(gòu)造方法還可通過傳入變量還決定調(diào)用方是使用公平鎖還是非公平鎖;4.2、Sync同步器1、AQS --> Sync ---> FairSync // 公平鎖 | |> NonfairSync // 非公平鎖 2、ReentrantLock內(nèi)的同步器都是通過Sync抽象接口來操作調(diào)用關(guān)系的,細(xì)看會發(fā)現(xiàn)基本上都是通過sync.xxx之類的這種調(diào)用方式的;4.3、lock()1、源碼: public void lock() { sync.lock(); } // FairSync 公平鎖調(diào)用方式 final void lock() { acquire(1); // 嘗試獲取獨占鎖 } // NonfairSync 非公平鎖調(diào)用方式 final void lock() { if (compareAndSetState(0, 1)) // 首先判斷state資源是否為0,如果恰巧為0則表明目前沒有線程占用鎖,則利用CAS占有鎖 setExclusiveOwnerThread(Thread.currentThread()); // 當(dāng)獨占鎖之后則將設(shè)置exclusiveOwnerThread為當(dāng)前線程 else acquire(1); // 若CAS占用鎖失敗的話,則再嘗試獲取獨占鎖 } 2、這里的區(qū)別就是非公平鎖在調(diào)用lock時首先檢測了是否通過CAS獲取鎖,發(fā)現(xiàn)鎖一旦空著的話,則搶先一步占為己有, 不管有沒有阻塞隊列,只要當(dāng)前線程來的時候發(fā)現(xiàn)state資源沒被占用那么當(dāng)前線程就搶先一步試一下CAS,CAS失敗了它才去排隊;4.4、acquire(int)1、源碼: public final void acquire(int arg) { if (!tryAcquire(arg) && // 嘗試獲取鎖資源,若獲取到資源的話則線程直接返回,此方法由AQS的具體子類實現(xiàn) acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) // 否則獲取資源失敗的話,那么就進(jìn)入等待隊列 selfInterrupt(); } 2、該方法是獨占模式下線程獲取state共享資源的入口,如果獲取到資源的話就返回,否則創(chuàng)建獨占模式結(jié)點加入阻塞隊列,直到獲取到共享資源; 3、而且這里需要加上自我中斷判斷,主要是因為線程在等待過程中被中斷的話,它是不響應(yīng)的,那么就只有等到線程獲取到資源后通過自我判斷將這個判斷后續(xù)補上; 4、獨占模式的該方法,正常情況下只要沒有獲取到鎖,該方法一直處于阻塞狀態(tài),獲取到了則跳出該方法區(qū);4.5、tryAcquire(int)1、公平鎖tryAcquire源碼: // FairSync 公平鎖的 tryAcquire 方法 protected final boolean tryAcquire(int acquires) { final Thread current = Thread.currentThread(); int c = getState(); // 獲取鎖資源的最新內(nèi)存值 if (c == 0) { // 當(dāng)state=0,說明鎖資源目前還沒有被任何線程被占用 if (!hasQueuedPredecessors() && // 檢查線程是否有阻塞隊列 compareAndSetState(0, acquires)) { // 如果沒有阻塞隊列,則通過CAS操作獲取鎖資源 setExclusiveOwnerThread(current); // 沒有阻塞隊列,且CAS又成功獲取鎖資源,則設(shè)置獨占線程對象為當(dāng)前線程 return true; // 返回標(biāo)志,告訴上層該線程已經(jīng)獲取到了鎖資源 } } // 執(zhí)行到此,鎖資源值不為0,說明已經(jīng)有線程正在占用這鎖資源 else if (current == getExclusiveOwnerThread()) { // 既然鎖已經(jīng)被占用,則看看占用鎖的線程是不是當(dāng)前線程 int nextc = c + acquires; // 如果占用的鎖的線程是當(dāng)前線程的話,則為重入鎖概念,狀態(tài)值做加1操作 // int類型值小于0,是因為該int類型的state狀態(tài)值溢出了,溢出了的話那得說明這個鎖有多難獲取啊,可能出問題了 if (nextc < 0) throw new Error("Maximum lock count exceeded"); setState(nextc); return true; // 返回成功標(biāo)志,告訴上層該線程已經(jīng)獲取到了鎖資源 } return false; // 返回失敗標(biāo)志,告訴上層該線程沒有獲取到鎖資源 } 2、非公平鎖tryAcquire源碼: // NonfairSync 非公平鎖的 tryAcquire 方法 protected final boolean tryAcquire(int acquires) { return nonfairTryAcquire(acquires); // 調(diào)用父類的非公平獲取鎖資源方法 } // NonfairSync 非公平鎖父類 Sync 類的 nonfairTryAcquire 方法 final boolean nonfairTryAcquire(int acquires) { final Thread current = Thread.currentThread(); int c = getState(); // 獲取鎖資源的最新內(nèi)存值 if (c == 0) { // 當(dāng)state=0,說明鎖資源目前還沒有被任何線程被占用 if (compareAndSetState(0, acquires)) { // 先不管三七二十一,先嘗試通過CAS操作獲取鎖資源 setExclusiveOwnerThread(current); // CAS一旦成功獲取鎖資源,則設(shè)置獨占線程對象為當(dāng)前線程 return true;// 返回成功標(biāo)志,告訴上層該線程已經(jīng)獲取到了鎖資源 } } // 執(zhí)行到此,鎖資源值不為0,說明已經(jīng)有線程正在占用這鎖資源 else if (current == getExclusiveOwnerThread()) { // 既然鎖已經(jīng)被占用,則看看占用鎖的線程是不是當(dāng)前線程 int nextc = c + acquires; // 如果占用的鎖的線程是當(dāng)前線程的話,則為重入鎖概念,狀態(tài)值做加1操作 // int類型值小于0,是因為該int類型的state狀態(tài)值溢出了,溢出了的話那得說明這個鎖有多難獲取啊,可能出問題了 if (nextc < 0) // overflow throw new Error("Maximum lock count exceeded"); setState(nextc); // return true; // 返回成功標(biāo)志,告訴上層該線程已經(jīng)獲取到了鎖資源 } return false; // 返回失敗標(biāo)志,告訴上層該線程沒有獲取到鎖資源 } 3、tryAcquire方法是AQS的子類實現(xiàn)的,也就是ReentrantLock的兩個靜態(tài)內(nèi)部類實現(xiàn)的,目的就是通過CAS嘗試獲取鎖資源, 獲取鎖資源成功則返回true,獲取鎖資源失敗則返回false;4.6、addWaiter(Node)1、源碼: /** * Creates and enqueues node for current thread and given mode. * * @param mode Node.EXCLUSIVE for exclusive, Node.SHARED for shared * @return the new node */ private Node addWaiter(Node mode) { // 按照給定的mode模式創(chuàng)建新的結(jié)點,模式有兩種:Node.EXCLUSIVE獨占模式、Node.SHARED共享模式; Node node = new Node(Thread.currentThread(), mode); // Try the fast path of enq; backup to full enq on failure Node pred = tail; // 將先隊尾結(jié)點賦值給臨時變量 if (pred != null) { // 如果pred不為空,說明該隊列已經(jīng)有結(jié)點了 node.prev = pred; if (compareAndSetTail(pred, node)) { // 通過CAS嘗試將node結(jié)點設(shè)置為隊尾結(jié)點 pred.next = node; return node; } } // 執(zhí)行到此,說明隊尾沒有元素,則進(jìn)入自旋首先設(shè)置頭結(jié)點,然后將此新建結(jié)點添加到隊尾 enq(node); // 進(jìn)入自旋添加node結(jié)點 return node; } 2、 addWaiter通過傳入不同的模式來創(chuàng)建新的結(jié)點嘗試加入到隊列尾部,如果由于并發(fā)導(dǎo)致添加結(jié)點到隊尾失敗的話那么就進(jìn)入自旋將結(jié)點加入隊尾;4.7、enq(Node)1、源碼: /** * Inserts node into queue, initializing if necessary. See picture above. * @param node the node to insert * @return node"s predecessor */ private Node enq(final Node node) { for (;;) { // 自旋的死循環(huán)操作方式 Node t = tail; // 因為是自旋方式,首次鏈表隊列tail肯定為空,但是后續(xù)鏈表有數(shù)據(jù)后就不會為空了 if (t == null) { // Must initialize if (compareAndSetHead(new Node())) // 隊列為空時,則創(chuàng)建一個空對象結(jié)點作為頭結(jié)點,無意思,可認(rèn)為傀儡結(jié)點 tail = head; // 空隊列的話,頭尾都指向同一個對象 } else { // 進(jìn)入 else 方法里面,說明鏈表隊列已經(jīng)有結(jié)點了 node.prev = t; // 因為存在并發(fā)操作,通過CAS嘗試將新加入的node結(jié)點設(shè)置為隊尾結(jié)點 if (compareAndSetTail(t, node)) { // 如果node設(shè)置隊尾結(jié)點成功,則將之前的舊的對象尾結(jié)點t的后繼結(jié)點指向node,node的前驅(qū)結(jié)點也設(shè)置為t t.next = node; return t; } } // 如果執(zhí)行到這里,說明上述兩個CAS操作任何一個失敗的話,該方法是不會放棄的,因為是自旋操作,再次循環(huán)繼續(xù)入隊 } } 2、enq通過自旋這種死循環(huán)的操作方式,來確保結(jié)點正確的添加到隊列尾部,通過CAS操作如果頭部為空則添加傀儡空結(jié)點,然后在循環(huán)添加隊尾結(jié)點;4.8、compareAndSetHead/compareAndSetTail1、源碼: /** * CAS head field. Used only by enq. */ private final boolean compareAndSetHead(Node update) { return unsafe.compareAndSwapObject(this, headOffset, null, update); } /** * CAS tail field. Used only by enq. */ private final boolean compareAndSetTail(Node expect, Node update) { return unsafe.compareAndSwapObject(this, tailOffset, expect, update); } 2、CAS操作,設(shè)置頭結(jié)點、尾結(jié)點;4.9、acquireQueued(Node, int)1、源碼: /** * Acquires in exclusive uninterruptible mode for thread already in * queue. Used by condition wait methods as well as acquire. * * @param node the node * @param arg the acquire argument * @return {@code true} if interrupted while waiting */ final boolean acquireQueued(final Node node, int arg) { boolean failed = true; try { boolean interrupted = false; for (;;) { // 自旋的死循環(huán)操作方式 final Node p = node.predecessor(); // 如果新建結(jié)點的前驅(qū)結(jié)點是頭結(jié)點 // 如果前驅(qū)結(jié)點為頭結(jié)點,那么該結(jié)點則是老二,僅次于老大,也希望嘗試去獲取一下鎖,萬一頭結(jié)點恰巧剛剛釋放呢?希望還是要有的,萬一實現(xiàn)了呢。。。 if (p == head && tryAcquire(arg)) { setHead(node); p.next = null; // help GC // 拿到鎖資源后,則該node結(jié)點升級做頭結(jié)點,且設(shè)置后繼結(jié)點指針為空,便于GC回收 failed = false; return interrupted; } if (shouldParkAfterFailedAcquire(p, node) && // 根據(jù)前驅(qū)結(jié)點看看是否需要休息一會兒 parkAndCheckInterrupt()) // 阻塞操作,正常情況下,獲取不到鎖,代碼就在該方法停止了,直到被喚醒 interrupted = true; // 如果執(zhí)行到這里,說明嘗試休息失敗了,因為是自旋操作,所以還會再次循環(huán)繼續(xù)操作判斷 } } finally { if (failed) cancelAcquire(node); } } 2、acquireQueued也是采用一個自旋的死循環(huán)操作方式,只有頭結(jié)點才能嘗試獲取鎖資源,其余的結(jié)點挨個挨個在那里等待修改,等待被喚醒,等待機(jī)會成為頭結(jié)點; 而新添加的node結(jié)點也自然逃不過如此命運,先看看是否頭結(jié)點,然后再看看是否能休息;4.10、shouldParkAfterFailedAcquire(Node, Node)1、源碼: /** * Checks and updates status for a node that failed to acquire. * Returns true if thread should block. This is the main signal * control in all acquire loops. Requires that pred == node.prev. * * @param pred node"s predecessor holding status * @param node the node * @return {@code true} if thread should block */ private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) { int ws = pred.waitStatus; // 獲取前驅(qū)結(jié)點的狀態(tài)值 if (ws == Node.SIGNAL) // 若前驅(qū)結(jié)點的狀態(tài)為SIGNAL狀態(tài)的話,那么該結(jié)點就不要想事了,直接返回true準(zhǔn)備休息 /* * This node has already set status asking a release * to signal it, so it can safely park. */ return true; if (ws > 0) { /* * Predecessor was cancelled. Skip over predecessors and * indicate retry. */ // 若前驅(qū)結(jié)點的狀態(tài)為CANCELLED狀態(tài)的話,那么就一直向前遍歷,直到找到一個不為CANCELLED狀態(tài)的結(jié)點 do { node.prev = pred = pred.prev; } while (pred.waitStatus > 0); pred.next = node; } else { /* * waitStatus must be 0 or PROPAGATE. Indicate that we * need a signal, but don"t park yet. Caller will need to * retry to make sure it cannot acquire before parking. */ // 剩下的結(jié)點狀態(tài),則設(shè)置其為SIGNAL狀態(tài),然后返回false標(biāo)志等外層循環(huán)再次判斷 compareAndSetWaitStatus(pred, ws, Node.SIGNAL); } return false; } 2、shouldParkAfterFailedAcquire主要是檢測前驅(qū)結(jié)點狀態(tài),前驅(qū)結(jié)點為SIGNAL的話,則新結(jié)點可以安安心心休息了; 如果前驅(qū)結(jié)點大于零,說明前驅(qū)結(jié)點處于CANCELLED狀態(tài),那么則以入?yún)red前驅(qū)為起點,一直往前找,直到找到最近一個正常等待狀態(tài)的結(jié)點; 如果前驅(qū)結(jié)點小于零,那么就將前驅(qū)結(jié)點設(shè)置為SIGNAL狀態(tài),然后返回false依賴acquireQueued的自旋再次判斷是否需要進(jìn)行休息;4.11、parkAndCheckInterrupt()1、源碼: /** * Convenience method to park and then check if interrupted * * @return {@code true} if interrupted */ private final boolean parkAndCheckInterrupt() { LockSupport.park(this); // 阻塞等待 return Thread.interrupted(); // 被喚醒后查看是否有被中斷過否? } 2、parkAndCheckInterrupt首先調(diào)用park讓線程進(jìn)入等待狀態(tài),然后當(dāng)park阻塞被喚醒后,再次檢測是否曾經(jīng)被中斷過; 而被喚醒有兩種情況,一個是利用unpark喚醒,一個是利用interrupt喚醒;4.12、unlock()1、源碼: public void unlock() { sync.release(1); // } 2、unlock釋放鎖資源,一般都是在finally中被調(diào)用,防止當(dāng)臨界區(qū)因為任何異常時怕鎖不被釋放; 而釋放鎖不像獲取鎖lock的實現(xiàn)多色多樣,沒有所謂公平或不公平,就是規(guī)規(guī)矩矩的釋放資源而已;4.13、release(int)1、源碼: /** * Releases in exclusive mode. Implemented by unblocking one or * more threads if {@link #tryRelease} returns true. * This method can be used to implement method {@link Lock#unlock}. * * @param arg the release argument. This value is conveyed to * {@link #tryRelease} but is otherwise uninterpreted and * can represent anything you like. * @return the value returned from {@link #tryRelease} */ public final boolean release(int arg) { if (tryRelease(arg)) { // 嘗試釋放鎖資源,此方法由AQS的具體子類實現(xiàn) Node h = head; if (h != null && h.waitStatus != 0) // 從頭結(jié)點開始,喚醒后繼結(jié)點 unparkSuccessor(h); // 踢出CANCELLED狀態(tài)結(jié)點,然后喚醒后繼結(jié)點 return true; } return false; } 2、release嘗試釋放鎖,并且有義務(wù)移除CANCELLED狀態(tài)的結(jié)點,還有義務(wù)喚醒后繼結(jié)點繼續(xù)運行獲取鎖資源;4.14、tryRelease(int)1、源碼: // NonfairSync 和 FairSync 的父類 Sync 類的 tryRelease 方法 protected final boolean tryRelease(int releases) { int c = getState() - releases; // 獲取鎖資源值并做減1操作 if (Thread.currentThread() != getExclusiveOwnerThread()) // 查看當(dāng)前線程是否和持有鎖的線程是不是同一個線程 // 正常情況下,需要釋放的線程肯定是持有鎖的線程,否則不就亂套了,肯定哪里出問題了,所以拋出異常 throw new IllegalMonitorStateException(); boolean free = false; if (c == 0) { // 若此時鎖資源值做減法操作后正好是0,則所有鎖資源已經(jīng)釋放干凈,因此持有鎖的變量也置為空 free = true; setExclusiveOwnerThread(null); } setState(c); // 若此時做減法操作還沒有歸零,那么這種情況就是那種重入鎖,需要重重釋放后才行 return free; } 2、tryRelease主要通過CAS操作對state鎖資源進(jìn)行減1操作;4.15、unparkSuccessor(Node)1、源碼: /** * Wakes up node"s successor, if one exists. * * @param node the node */ private void unparkSuccessor(Node node) { /* * If status is negative (i.e., possibly needing signal) try * to clear in anticipation of signalling. It is OK if this * fails or if status is changed by waiting thread. */ // 該node一般都是傳入head進(jìn)來,也就是說,需要釋放頭結(jié)點,也就是當(dāng)前結(jié)點需要釋放鎖操作,順便喚醒后繼結(jié)點 int ws = node.waitStatus; if (ws < 0) // 若結(jié)點狀態(tài)值小于0,則歸零處理,通過CAS歸零,允許失敗,但是不管怎么著,仍然要往下走去喚醒后繼結(jié)點 compareAndSetWaitStatus(node, ws, 0); /* * Thread to unpark is held in successor, which is normally * just the next node. But if cancelled or apparently null, * traverse backwards from tail to find the actual * non-cancelled successor. */ Node s = node.next; // 取出后繼結(jié)點,這個時候一般都是Head后面的一個結(jié)點,所以一般都是老二 if (s == null || s.waitStatus > 0) { // 若后繼結(jié)點為空或者后繼結(jié)點已經(jīng)處于CANCELLED狀態(tài)的話 s = null; // 那么從隊尾向前遍歷,直到找到一個小于等于0的結(jié)點 // 這里為什么要從隊尾向前尋找? // * 因為在這個隊列中,任何一個結(jié)點都有可能被中斷,只是有可能,并不代表絕對的,但有一點是確定的, // * 被中斷的結(jié)點會將結(jié)點的狀態(tài)設(shè)置為CANCELLED狀態(tài),標(biāo)識這個結(jié)點在將來的某個時刻會被踢出; // * 踢出隊列的規(guī)則很簡單,就是該結(jié)點的前驅(qū)結(jié)點不會指向它,而是會指向它的后面的一個非CANCELLED狀態(tài)的結(jié)點; // * 而這個將被踢出的結(jié)點,它的next指針將會指向它自己; // * 所以設(shè)想一下,如果我們從head往后找,一旦發(fā)現(xiàn)這么一個處于CANCELLED狀態(tài)的結(jié)點,那么for循環(huán)豈不是就是死循環(huán)了; // * 但是所有的這些結(jié)點當(dāng)中,它們的prev前驅(qū)結(jié)點還是沒有被誰動過,所以從tail結(jié)點向前遍歷最穩(wěn)妥 for (Node t = tail; t != null && t != node; t = t.prev) if (t.waitStatus <= 0) s = t; } if (s != null) LockSupport.unpark(s.thread); // 喚醒線程 } 2、unparkSuccessor主要是踢出CANCELLED狀態(tài)結(jié)點,然后喚醒后繼結(jié)點; 但是這個喚醒的后繼結(jié)點為空的話,那么則從隊尾一直向前循環(huán)查找小于等于零狀態(tài)的結(jié)點并調(diào)用unpark喚醒;五、總結(jié)1、分析了這么多,感覺是不是有一種豁然開朗的感覺,原來大家傳的神乎其神的AQS是不是沒有想象中那么難以理解; 2、在這里我簡要總結(jié)一下AQS的流程的一些特性: ? 關(guān)鍵獲取鎖、釋放鎖操作由AQS子類實現(xiàn):acquire-release、acquireShared-releaseShared; ? 維護(hù)了一個FIFO鏈表結(jié)構(gòu)的隊列,通過自旋方式將新結(jié)點添加到隊尾; ? 添加結(jié)點時會從前驅(qū)結(jié)點向前遍歷,跳過那些處于CANCELLED狀態(tài)的結(jié)點; ? 釋放結(jié)點時會從隊尾向前遍歷,踢出CANCELLED狀態(tài)的結(jié)點,然后喚醒后繼結(jié)點; 3、其實當(dāng)了解了AQS后,這里以ReentrantLock為載體分析了一下,那么再去分析CountDownLatch、Semaphore、ReentrantReadWriteLock等那些集成AQS而實現(xiàn)不同功能的模塊就會順利很多;六、下載地址https://gitee.com/ylimhhmily/SpringCloudTutorial.git
SpringCloudTutorial交流QQ群: 235322432
SpringCloudTutorial交流微信群: 微信溝通群二維碼圖片鏈接
歡迎關(guān)注,您的肯定是對我最大的支持!!!
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/71084.html
摘要:原理剖析第篇工作原理分析一大致介紹關(guān)于多線程競爭鎖方面,大家都知道有個和,也正是這兩個東西才引申出了大量的線程安全類,鎖類等功能而隨著現(xiàn)在的硬件廠商越來越高級,在硬件層面提供大量并發(fā)原語給我們層面的開發(fā)帶來了莫大的利好本章節(jié)就和大家分享分 原理剖析(第 004 篇)CAS工作原理分析 - 一、大致介紹 1、關(guān)于多線程競爭鎖方面,大家都知道有個CAS和AQS,也正是這兩個東西才引申出了大...
摘要:表示的是兩個,當(dāng)其中任意一個計算完并發(fā)編程之是線程安全并且高效的,在并發(fā)編程中經(jīng)??梢娝氖褂?,在開始分析它的高并發(fā)實現(xiàn)機(jī)制前,先講講廢話,看看它是如何被引入的。電商秒殺和搶購,是兩個比較典型的互聯(lián)網(wǎng)高并發(fā)場景。 干貨:深度剖析分布式搜索引擎設(shè)計 分布式,高可用,和機(jī)器學(xué)習(xí)一樣,最近幾年被提及得最多的名詞,聽名字多牛逼,來,我們一步一步來擊破前兩個名詞,今天我們首先來說說分布式。 探究...
摘要:開始獲取鎖終于輪到出場了,的調(diào)用過程和完全一樣,同樣拿不到鎖,然后加入到等待隊列隊尾然后,在阻塞前需要把前驅(qū)結(jié)點的狀態(tài)置為,以確保將來可以被喚醒至此,的執(zhí)行也暫告一段落了安心得在等待隊列中睡覺。 showImg(https://segmentfault.com/img/remote/1460000016012467); 本文首發(fā)于一世流云的專欄:https://segmentfault...
閱讀 3592·2021-11-18 13:20
閱讀 2738·2021-10-15 09:40
閱讀 1765·2021-10-11 10:58
閱讀 2130·2021-09-27 13:36
閱讀 2602·2021-09-07 10:06
閱讀 1862·2021-08-11 11:21
閱讀 1435·2019-08-29 17:04
閱讀 2090·2019-08-29 14:06