偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

深入JVM鎖機(jī)制之二:Lock

開發(fā) 后端
前文(深入JVM鎖機(jī)制-synchronized)分析了JVM中的synchronized實(shí)現(xiàn),本文繼續(xù)分析JVM中的另一種鎖Lock的實(shí)現(xiàn)。與synchronized不同的是,Lock完全用Java寫成,在java這個(gè)層面是無關(guān)JVM實(shí)現(xiàn)的。

前文(深入JVM鎖機(jī)制-synchronized)分析了JVM中的synchronized實(shí)現(xiàn),本文繼續(xù)分析JVM中的另一種鎖Lock的實(shí)現(xiàn)。與synchronized不同的是,Lock完全用Java寫成,在java這個(gè)層面是無關(guān)JVM實(shí)現(xiàn)的。

在java.util.concurrent.locks包中有很多Lock的實(shí)現(xiàn)類,常用的有ReentrantLock、ReadWriteLock(實(shí)現(xiàn)類ReentrantReadWriteLock),其實(shí)現(xiàn)都依賴java.util.concurrent.AbstractQueuedSynchronizer類,實(shí)現(xiàn)思路都大同小異,因此我們以ReentrantLock作為講解切入點(diǎn)。

1. ReentrantLock的調(diào)用過程

經(jīng)過觀察ReentrantLock把所有Lock接口的操作都委派到一個(gè)Sync類上,該類繼承了AbstractQueuedSynchronizer:

  1. static abstract class Sync extends AbstractQueuedSynchronizer  

Sync又有兩個(gè)子類:

  1. final static class NonfairSync extends Sync
  1. final static class FairSync extends Sync 

顯然是為了支持公平鎖和非公平鎖而定義,默認(rèn)情況下為非公平鎖。

先理一下Reentrant.lock()方法的調(diào)用過程(默認(rèn)非公平鎖):

這些討厭的Template模式導(dǎo)致很難直觀的看到整個(gè)調(diào)用過程,其實(shí)通過上面調(diào)用過程及AbstractQueuedSynchronizer的注釋可以發(fā)現(xiàn),AbstractQueuedSynchronizer中抽象了絕大多數(shù)Lock的功能,而只把tryAcquire方法延遲到子類中實(shí)現(xiàn)。tryAcquire方法的語義在于用具體子類判斷請(qǐng)求線程是否可以獲得鎖,無論成功與否AbstractQueuedSynchronizer都將處理后面的流程。

2. 鎖實(shí)現(xiàn)(加鎖)

簡(jiǎn)單說來,AbstractQueuedSynchronizer會(huì)把所有的請(qǐng)求線程構(gòu)成一個(gè)CLH隊(duì)列,當(dāng)一個(gè)線程執(zhí)行完畢(lock.unlock())時(shí)會(huì)激活自己的后繼節(jié)點(diǎn),但正在執(zhí)行的線程并不在隊(duì)列中,而那些等待執(zhí)行的線程全部處于阻塞狀態(tài),經(jīng)過調(diào)查線程的顯式阻塞是通過調(diào)用LockSupport.park()完成,而LockSupport.park()則調(diào)用sun.misc.Unsafe.park()本地方法,再進(jìn)一步,HotSpot在Linux中中通過調(diào)用pthread_mutex_lock函數(shù)把線程交給系統(tǒng)內(nèi)核進(jìn)行阻塞。

該隊(duì)列如圖:

與synchronized相同的是,這也是一個(gè)虛擬隊(duì)列,不存在隊(duì)列實(shí)例,僅存在節(jié)點(diǎn)之間的前后關(guān)系。令人疑惑的是為什么采用CLH隊(duì)列呢?原生的CLH隊(duì)列是用于自旋鎖,但Doug Lea把其改造為阻塞鎖。

當(dāng)有線程競(jìng)爭(zhēng)鎖時(shí),該線程會(huì)首先嘗試獲得鎖,這對(duì)于那些已經(jīng)在隊(duì)列中排隊(duì)的線程來說顯得不公平,這也是非公平鎖的由來,與synchronized實(shí)現(xiàn)類似,這樣會(huì)極大提高吞吐量。

如果已經(jīng)存在Running線程,則新的競(jìng)爭(zhēng)線程會(huì)被追加到隊(duì)尾,具體是采用基于CAS的Lock-Free算法,因?yàn)榫€程并發(fā)對(duì)Tail調(diào)用CAS可能會(huì)導(dǎo)致其他線程CAS失敗,解決辦法是循環(huán)CAS直至成功。AbstractQueuedSynchronizer的實(shí)現(xiàn)非常精巧,令人嘆為觀止,不入細(xì)節(jié)難以完全領(lǐng)會(huì)其精髓,下面詳細(xì)說明實(shí)現(xiàn)過程:
 

2.1 Sync.nonfairTryAcquire

nonfairTryAcquire方法將是lock方法間接調(diào)用的***個(gè)方法,每次請(qǐng)求鎖時(shí)都會(huì)首先調(diào)用該方法。

  1. final boolean nonfairTryAcquire(int acquires) {   
  2.     final Thread current = Thread.currentThread();   
  3.     int c = getState();   
  4.     if (c == 0) {   
  5.         if (compareAndSetState(0, acquires)) {   
  6.             setExclusiveOwnerThread(current);   
  7.             return true;   
  8.         }   
  9.     }   
  10.     else if (current == getExclusiveOwnerThread()) {   
  11.         int nextc = c + acquires;   
  12.         if (nextc < 0// overflow   
  13.             throw new Error("Maximum lock count exceeded");   
  14.         setState(nextc);   
  15.         return true;   
  16.     }   
  17.     return false;   
  18. }   

該方法會(huì)首先判斷當(dāng)前狀態(tài),如果c==0說明沒有線程正在競(jìng)爭(zhēng)該鎖,如果不c !=0 說明有線程正擁有了該鎖。

如果發(fā)現(xiàn)c==0,則通過CAS設(shè)置該狀態(tài)值為acquires,acquires的初始調(diào)用值為1,每次線程重入該鎖都會(huì)+1,每次unlock都會(huì)-1,但為0時(shí)釋放鎖。如果CAS設(shè)置成功,則可以預(yù)計(jì)其他任何線程調(diào)用CAS都不會(huì)再成功,也就認(rèn)為當(dāng)前線程得到了該鎖,也作為Running線程,很顯然這個(gè)Running線程并未進(jìn)入等待隊(duì)列。

如果c !=0 但發(fā)現(xiàn)自己已經(jīng)擁有鎖,只是簡(jiǎn)單地++acquires,并修改status值,但因?yàn)闆]有競(jìng)爭(zhēng),所以通過setStatus修改,而非CAS,也就是說這段代碼實(shí)現(xiàn)了偏向鎖的功能,并且實(shí)現(xiàn)的非常漂亮。

2.2 AbstractQueuedSynchronizer.addWaiter

addWaiter方法負(fù)責(zé)把當(dāng)前無法獲得鎖的線程包裝為一個(gè)Node添加到隊(duì)尾:

  1. private Node addWaiter(Node mode) {   
  2.     Node node = new Node(Thread.currentThread(), mode);   
  3.     // Try the fast path of enq; backup to full enq on failure   
  4.     Node pred = tail;   
  5.     if (pred != null) {   
  6.         node.prev = pred;   
  7.         if (compareAndSetTail(pred, node)) {   
  8.             pred.next = node;   
  9.             return node;   
  10.         }   
  11.     }   
  12.     enq(node);   
  13.     return node;   
  14. }   
其中參數(shù)mode是獨(dú)占鎖還是共享鎖,默認(rèn)為null,獨(dú)占鎖。追加到隊(duì)尾的動(dòng)作分兩步:
  1. 如果當(dāng)前隊(duì)尾已經(jīng)存在(tail!=null),則使用CAS把當(dāng)前線程更新為Tail。
  2. 如果當(dāng)前Tail為null或則線程調(diào)用CAS設(shè)置隊(duì)尾失敗,則通過enq方法繼續(xù)設(shè)置Tail。
下面是enq方法:
  1. private Node enq(final Node node) {   
  2.     for (;;) {   
  3.         Node t = tail;   
  4.         if (t == null) { // Must initialize   
  5.             Node h = new Node(); // Dummy header   
  6.             h.next = node;   
  7.             node.prev = h;   
  8.             if (compareAndSetHead(h)) {   
  9.                 tail = node;   
  10.                 return h;   
  11.             }   
  12.         }   
  13.         else {   
  14.             node.prev = t;   
  15.             if (compareAndSetTail(t, node)) {   
  16.                 t.next = node;   
  17.                 return t;   
  18.             }   
  19.         }   
  20.     }   
  21. }   

該方法就是循環(huán)調(diào)用CAS,即使有高并發(fā)的場(chǎng)景,無限循環(huán)將會(huì)最終成功把當(dāng)前線程追加到隊(duì)尾(或設(shè)置隊(duì)頭)??偠灾?,addWaiter的目的就是通過CAS把當(dāng)前現(xiàn)在追加到隊(duì)尾,并返回包裝后的Node實(shí)例。

把線程要包裝為Node對(duì)象的主要原因,除了用Node構(gòu)造供虛擬隊(duì)列外,還用Node包裝了各種線程狀態(tài),這些狀態(tài)被精心設(shè)計(jì)為一些數(shù)字值:

◆ SIGNAL(-1) :線程的后繼線程正/已被阻塞,當(dāng)該線程release或cancel時(shí)要重新這個(gè)后繼線程(unpark)。

◆ CANCELLED(1):因?yàn)槌瑫r(shí)或中斷,該線程已經(jīng)被取消。

◆ CONDITION(-2):表明該線程被處于條件隊(duì)列,就是因?yàn)檎{(diào)用了Condition.await而被阻塞。

◆ PROPAGATE(-3):傳播共享鎖。

◆ 0:0代表無狀態(tài)。

2.3 AbstractQueuedSynchronizer.acquireQueued

acquireQueued的主要作用是把已經(jīng)追加到隊(duì)列的線程節(jié)點(diǎn)(addWaiter方法返回值)進(jìn)行阻塞,但阻塞前又通過tryAccquire重試是否能獲得鎖,如果重試成功能則無需阻塞,直接返回

  1. final boolean acquireQueued(final Node node, int arg) {   
  2.     try {   
  3.         boolean interrupted = false;   
  4.         for (;;) {   
  5.             final Node p = node.predecessor();   
  6.             if (p == head && tryAcquire(arg)) {   
  7.                 setHead(node);   
  8.                 p.next = null// help GC   
  9.                 return interrupted;   
  10.             }   
  11.             if (shouldParkAfterFailedAcquire(p, node) &&   
  12.                 parkAndCheckInterrupt())   
  13.                 interrupted = true;   
  14.         }   
  15.     } catch (RuntimeException ex) {   
  16.         cancelAcquire(node);   
  17.         throw ex;   
  18.     }   
  19. }   

仔細(xì)看看這個(gè)方法是個(gè)無限循環(huán),感覺如果p == head && tryAcquire(arg)條件不滿足循環(huán)將永遠(yuǎn)無法結(jié)束,當(dāng)然不會(huì)出現(xiàn)死循環(huán),奧秘在于第12行的parkAndCheckInterrupt會(huì)把當(dāng)前線程掛起,從而阻塞住線程的調(diào)用棧。

  1. private final boolean parkAndCheckInterrupt() { 
  2.     LockSupport.park(this); 
  3.     return Thread.interrupted(); 

如前面所述,LockSupport.park最終把線程交給系統(tǒng)(Linux)內(nèi)核進(jìn)行阻塞。當(dāng)然也不是馬上把請(qǐng)求不到鎖的線程進(jìn)行阻塞,還要檢查該線程的狀態(tài),比如如果該線程處于Cancel狀態(tài)則沒有必要,具體的檢查在shouldParkAfterFailedAcquire中:

  1.   private static boolean shouldParkAfterFailedAcquire(Node pred, Node node) {   
  2.       int ws = pred.waitStatus;   
  3.       if (ws == Node.SIGNAL)   
  4.           /*  
  5.            * This node has already set status asking a release  
  6.            * to signal it, so it can safely park  
  7.            */   
  8.           return true;   
  9.       if (ws > 0) {   
  10.           /*  
  11.            * Predecessor was cancelled. Skip over predecessors and  
  12.            * indicate retry.  
  13.            */   
  14.    do {   
  15. node.prev = pred = pred.prev;   
  16.    } while (pred.waitStatus > 0);   
  17.    pred.next = node;   
  18.       } else {   
  19.           /*  
  20.            * waitStatus must be 0 or PROPAGATE. Indicate that we  
  21.            * need a signal, but don't park yet. Caller will need to  
  22.            * retry to make sure it cannot acquire before parking.   
  23.            */   
  24.           compareAndSetWaitStatus(pred, ws, Node.SIGNAL);   
  25.       }    
  26.       return false;   
  27.   }   

檢查原則在于:

◆ 規(guī)則1:如果前繼的節(jié)點(diǎn)狀態(tài)為SIGNAL,表明當(dāng)前節(jié)點(diǎn)需要unpark,則返回成功,此時(shí)acquireQueued方法的第12行(parkAndCheckInterrupt)將導(dǎo)致線程阻塞。

◆ 規(guī)則2:如果前繼節(jié)點(diǎn)狀態(tài)為CANCELLED(ws>0),說明前置節(jié)點(diǎn)已經(jīng)被放棄,則回溯到一個(gè)非取消的前繼節(jié)點(diǎn),返回false,acquireQueued方法的無限循環(huán)將遞歸調(diào)用該方法,直至規(guī)則1返回true,導(dǎo)致線程阻塞。

◆ 規(guī)則3:如果前繼節(jié)點(diǎn)狀態(tài)為非SIGNAL、非CANCELLED,則設(shè)置前繼的狀態(tài)為SIGNAL,返回false后進(jìn)入acquireQueued的無限循環(huán),與規(guī)則2同。

總體看來,shouldParkAfterFailedAcquire就是靠前繼節(jié)點(diǎn)判斷當(dāng)前線程是否應(yīng)該被阻塞,如果前繼節(jié)點(diǎn)處于CANCELLED狀態(tài),則順便刪除這些節(jié)點(diǎn)重新構(gòu)造隊(duì)列。

至此,鎖住線程的邏輯已經(jīng)完成,下面討論解鎖的過程。

3. 解 鎖

請(qǐng)求鎖不成功的線程會(huì)被掛起在acquireQueued方法的第12行,12行以后的代碼必須等線程被解鎖鎖才能執(zhí)行,假如被阻塞的線程得到解鎖,則執(zhí)行第13行,即設(shè)置interrupted = true,之后又進(jìn)入無限循環(huán)。

從無限循環(huán)的代碼可以看出,并不是得到解鎖的線程一定能獲得鎖,必須在第6行中調(diào)用tryAccquire重新競(jìng)爭(zhēng),因?yàn)殒i是非公平的,有可能被新加入的線程獲得,從而導(dǎo)致剛被喚醒的線程再次被阻塞,這個(gè)細(xì)節(jié)充分體現(xiàn)了“非公平”的精髓。通過之后將要介紹的解鎖機(jī)制會(huì)看到,***個(gè)被解鎖的線程就是Head,因此p == head的判斷基本都會(huì)成功。

至此可以看到,把tryAcquire方法延遲到子類中實(shí)現(xiàn)的做法非常精妙并具有極強(qiáng)的可擴(kuò)展性,令人嘆為觀止!當(dāng)然精妙的不是這個(gè)Templae設(shè)計(jì)模式,而是Doug Lea對(duì)鎖結(jié)構(gòu)的精心布局。

解鎖代碼相對(duì)簡(jiǎn)單,主要體現(xiàn)在AbstractQueuedSynchronizer.release和Sync.tryRelease方法中:

class AbstractQueuedSynchronizer

  1. public final boolean release(int arg) {   
  2.     if (tryRelease(arg)) {   
  3.         Node h = head;   
  4.         if (h != null && h.waitStatus != 0)   
  5.             unparkSuccessor(h);   
  6.         return true;   
  7.     }   
  8.     return false;   
  9. }   

class Sync

  1. protected final boolean tryRelease(int releases) {   
  2.     int c = getState() - releases;   
  3.     if (Thread.currentThread() != getExclusiveOwnerThread())   
  4.         throw new IllegalMonitorStateException();   
  5.     boolean free = false;   
  6.     if (c == 0) {   
  7.         free = true;   
  8.         setExclusiveOwnerThread(null);   
  9.     }   
  10.     setState(c);   
  11.     return free;   
  12. }   

tryRelease與tryAcquire語義相同,把如何釋放的邏輯延遲到子類中。tryRelease語義很明確:如果線程多次鎖定,則進(jìn)行多次釋放,直至status==0則真正釋放鎖,所謂釋放鎖即設(shè)置status為0,因?yàn)闊o競(jìng)爭(zhēng)所以沒有使用CAS。

release的語義在于:如果可以釋放鎖,則喚醒隊(duì)列***個(gè)線程(Head),具體喚醒代碼如下:

  1. private void unparkSuccessor(Node node) {   
  2.     /*  
  3.      * If status is negative (i.e., possibly needing signal) try  
  4.      * to clear in anticipation of signalling. It is OK if this  
  5.      * fails or if status is changed by waiting thread.  
  6.      */   
  7.     int ws = node.waitStatus;   
  8.     if (ws < 0)   
  9.         compareAndSetWaitStatus(node, ws, 0);    
  10.    
  11.     /*  
  12.      * Thread to unpark is held in successor, which is normally  
  13.      * just the next node.  But if cancelled or apparently null,  
  14.      * traverse backwards from tail to find the actual  
  15.      * non-cancelled successor.  
  16.      */   
  17.     Node s = node.next;   
  18.     if (s == null || s.waitStatus > 0) {   
  19.         s = null;   
  20.         for (Node t = tail; t != null && t != node; t = t.prev)   
  21.             if (t.waitStatus <= 0)   
  22.                 s = t;   
  23.     }   
  24.     if (s != null)   
  25.         LockSupport.unpark(s.thread);   
  26. }   

這段代碼的意思在于找出***個(gè)可以u(píng)npark的線程,一般說來head.next == head,Head就是***個(gè)線程,但Head.next可能被取消或被置為null,因此比較穩(wěn)妥的辦法是從后往前找***個(gè)可用線程。貌似回溯會(huì)導(dǎo)致性能降低,其實(shí)這個(gè)發(fā)生的幾率很小,所以不會(huì)有性能影響。之后便是通知系統(tǒng)內(nèi)核繼續(xù)該線程,在Linux下是通過pthread_mutex_unlock完成。之后,被解鎖的線程進(jìn)入上面所說的重新競(jìng)爭(zhēng)狀態(tài)。

4. Lock VS Synchronized

AbstractQueuedSynchronizer通過構(gòu)造一個(gè)基于阻塞的CLH隊(duì)列容納所有的阻塞線程,而對(duì)該隊(duì)列的操作均通過Lock-Free(CAS)操作,但對(duì)已經(jīng)獲得鎖的線程而言,ReentrantLock實(shí)現(xiàn)了偏向鎖的功能。

synchronized的底層也是一個(gè)基于CAS操作的等待隊(duì)列,但JVM實(shí)現(xiàn)的更精細(xì),把等待隊(duì)列分為ContentionList和EntryList,目的是為了降低線程的出列速度;當(dāng)然也實(shí)現(xiàn)了偏向鎖,從數(shù)據(jù)結(jié)構(gòu)來說二者設(shè)計(jì)沒有本質(zhì)區(qū)別。但synchronized還實(shí)現(xiàn)了自旋鎖,并針對(duì)不同的系統(tǒng)和硬件體系進(jìn)行了優(yōu)化,而Lock則完全依靠系統(tǒng)阻塞掛起等待線程。

當(dāng)然Lock比synchronized更適合在應(yīng)用層擴(kuò)展,可以繼承AbstractQueuedSynchronizer定義各種實(shí)現(xiàn),比如實(shí)現(xiàn)讀寫鎖(ReadWriteLock),公平或不公平鎖;同時(shí),Lock對(duì)應(yīng)的Condition也比wait/notify要方便的多、靈活的多。

原文鏈接:http://blog.csdn.net/chen77716/article/details/6641477

【編輯推薦】

  1. 深入JVM鎖機(jī)制之一:synchronized
  2. JVM優(yōu)化引起的邏輯錯(cuò)誤
  3. JVM中可生成的***Thread數(shù)量
  4. Eclipse推JVM語言Xtend 繼續(xù)擁抱Java
  5. 用JavaScript編寫JVM可成功運(yùn)行Java程序
責(zé)任編輯:林師授 來源: chen77716的博客
相關(guān)推薦

2011-11-28 12:31:20

JavaJVM

2010-11-22 14:18:32

MySQL鎖機(jī)制

2023-10-13 13:30:00

MySQL鎖機(jī)制

2021-09-24 08:10:40

Java 語言 Java 基礎(chǔ)

2024-07-08 12:51:05

2012-05-30 11:11:42

HTML5

2023-10-31 10:51:56

MySQLMVCC并發(fā)性

2024-12-02 09:01:23

Java虛擬機(jī)內(nèi)存

2025-05-21 10:05:00

C++11多線程編程

2021-06-28 10:51:55

Redisson分布式鎖Watchdog

2011-08-24 09:30:29

JavaJVM

2024-06-12 14:03:31

MySQLInnoDB

2021-07-09 06:48:31

ZooKeeperCurator源碼

2010-09-27 08:38:49

JVM堆JVM棧

2022-04-29 11:39:28

MySQL幻讀Gap Lock

2025-04-24 10:56:01

MySQLInnoDB數(shù)據(jù)庫(kù)鎖

2010-09-26 16:55:31

JVM學(xué)習(xí)筆記

2024-11-13 10:16:37

2023-10-31 16:00:51

類加載機(jī)制Java

2017-09-20 08:48:09

JVM內(nèi)存結(jié)構(gòu)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)