ReenTrantLock源碼淺析

## ReenTrantLock是什么

ReenTrantLock是juc包下的一個經(jīng)典的互斥鎖,也是 **可重入鎖**(即當前線程在已經(jīng)獲取改鎖后重復(fù)執(zhí)行獲取鎖操作時不會引起死鎖,并且不需要執(zhí)行獲取鎖的操作),ReenTrantLock是基于AQS來實現(xiàn)的(PS:注意是基于不是繼承,看到網(wǎng)上有的同學會說發(fā)現(xiàn)ReentrantLock并沒有繼承AQS有所疑問),如果有對AQS不了解的同學可以查看我之前的文章[淺析AQS(1)---獨占鎖以及共享鎖的實現(xiàn)](http://blog.xiaoazhai.com/aqs/)以及[淺析AQS (二)--condition的實現(xiàn)](http://blog.xiaoazhai.com/aqs2/),

<!--more-->

首先ReenTrantLock是默認的非公平鎖,但是也有公平鎖的實現(xiàn),需要在構(gòu)造方法中傳入是否公平的Boolean值來標志,需要注意的是,此處的公平與非公平只針對lock方法,而tryLock方法是指定非公平的.并不受限制

ReenTrantLock實現(xiàn)并發(fā)控制是依靠內(nèi)部類Sync及其兩個子類FairSync和NoFairSync,顧名思義這兩個子類是提供公平鎖以及非公平鎖的實現(xiàn),而Sync則是繼承了AQS的具體實現(xiàn)類,接下來我們結(jié)合源碼具體分析

## Sync內(nèi)部類

首先查看一下Sync的源碼

```java

? ? abstract static class Sync extends AbstractQueuedSynchronizer {

? ? ? ? private static final long serialVersionUID = -5179523762034025860L;

? ? ????//抽象lock方法 根據(jù)公平與非公平實現(xiàn)執(zhí)行不同的操作

? ? ? ? abstract void lock();

????????//非公平嘗試獲取鎖

? ? ? ? final boolean nonfairTryAcquire(int acquires) {

? ? ? ? ? ? final Thread current = Thread.currentThread();

? ? ? ? ? ? int c = getState();

? ? ? ? ? ? if (c == 0) {

? ? ? ? ? ? ? ? //如果當前鎖沒有被人獲取過則直接嘗試獲取鎖

? ? ? ? ? ? ? ? if (compareAndSetState(0, acquires)) {

? ? ? ? ? ? ? ? ? ? //設(shè)置當前持有鎖的線程

? ? ? ? ? ? ? ? ? ? setExclusiveOwnerThread(current);

? ? ? ? ? ? ? ? ? ? return true;

? ? ? ? ? ? ? ? }

? ? ? ? ? ? }

? ? ? ? ? ? else if (current == getExclusiveOwnerThread()) {

? ? ? ? ? ? ? ? //如果是當前線程獲取了當前鎖則直接重入

? ? ? ? ? ? ? ? int nextc = c + acquires;

? ? ? ? ? ? ? ? if (nextc < 0) // overflow

? ? ? ? ? ? ? ? ? ? throw new Error("Maximum lock count exceeded");

? ? ? ? ? ? ? ? setState(nextc);

? ? ? ? ? ? ? ? return true;

? ? ? ? ? ? }

? ? ? ? ? ? return false;

? ? ? ? }

????????//嘗試釋放鎖

? ? ? ? protected final boolean tryRelease(int releases) {

? ? ? ? ? ? int c = getState() - releases;

? ? ? ? ? ? if (Thread.currentThread() != getExclusiveOwnerThread())

? ? ? ? ? ? ? ? throw new IllegalMonitorStateException();

? ? ? ? ? ? boolean free = false;

? ? ? ? ? ? if (c == 0) {

? ? ? ? ? ? ? ? free = true;

? ? ? ? ? ? ? ? setExclusiveOwnerThread(null);

? ? ? ? ? ? }

? ? ? ? ? ? setState(c);

? ? ? ? ? ? return free;

? ? ? ? }

????????// 是否持有當前鎖

? ? ? ? protected final boolean isHeldExclusively() {

? ? ? ? ? ? return getExclusiveOwnerThread() == Thread.currentThread();

? ? ? ? }

????????//新建condition對象

? ? ? ? final ConditionObject newCondition() {

? ? ? ? ? ? return new ConditionObject();

? ? ? ? }

????????//獲取鎖當前的持有線程

? ? ? ? final Thread getOwner() {

? ? ? ? ? ? return getState() == 0 ? null : getExclusiveOwnerThread();

? ? ? ? }

????????//當前線程獲取到了

? ? ? ? final int getHoldCount() {

? ? ? ? ? ? return isHeldExclusively() ? getState() : 0;

? ? ? ? }

? ? ? ? final boolean isLocked() {

? ? ? ? ? ? return getState() != 0;

? ? ? ? }

? ? }

```

在Sync對象中,重寫了tryRelease方法,將釋放鎖的邏輯統(tǒng)一,并且定義了nonfairTryAcquire非公平獲取鎖的方法,這里為什么要將該方法定義在父類是因為無論是公平還是非公平鎖,tryLock方法中都是采用非公平獲取的方式,而無論維護的Sync對象時公平的還是非公平的,我們都需要一個可以非公平獲取鎖的方式

在nonfairTryAcquire我們可以看到,如果當前鎖的state為0即當前鎖沒人持有時則采用cas的方式獲取鎖,然后將持有現(xiàn)成設(shè)置為當前線程,而第二個判斷則是可重入鎖的實現(xiàn),即判斷持有鎖的是不是當前線程,如果是則直接獲取鎖

關(guān)于ConditionObject的實現(xiàn)可以查看我以前的文章[淺析AQS (二)--condition的實現(xiàn)](http://blog.xiaoazhai.com/aqs2/),

## FairSync與NoFairSync

下面放一下公平與非公平鎖的具體實現(xiàn)

```java

static final class NonfairSync extends Sync {

? ? ? ? private static final long serialVersionUID = 7316153563782823691L;


? ? ? ? final void lock() {

? ? ? ? ? ? //嘗試獲取鎖

? ? ? ? ? ? if (compareAndSetState(0, 1))

? ? ? ? ? ? ? ? setExclusiveOwnerThread(Thread.currentThread());

? ? ? ? ? ? else

? ? ? ? ? ? ? ? acquire(1);

? ? ? ? }

? ? ? ? protected final boolean tryAcquire(int acquires) {

? ? ? ? ? ? return nonfairTryAcquire(acquires);

? ? ? ? }

? ? }

? ? static final class FairSync extends Sync {

? ? ? ? private static final long serialVersionUID = -3000897897090466540L;

? ? ? ? final void lock() {

? ? ? ? ? ? acquire(1);

? ? ? ? }

? ? ? ? protected final boolean tryAcquire(int acquires) {

? ? ? ? ? ? final Thread current = Thread.currentThread();

? ? ? ? ? ? int c = getState();

? ? ? ? ? ? if (c == 0) {

? ? ? ? ? ? ? ? //判斷是否有在隊列中阻塞的線程,或者當前線程是否為隊列的頭

? ? ? ? ? ? ? ? if (!hasQueuedPredecessors() &&

? ? ? ? ? ? ? ? ? ? compareAndSetState(0, acquires)) {

? ? ? ? ? ? ? ? ? ? setExclusiveOwnerThread(current);

? ? ? ? ? ? ? ? ? ? return true;

? ? ? ? ? ? ? ? }

? ? ? ? ? ? }

? ? ? ? ? ? else if (current == getExclusiveOwnerThread()) {

? ? ? ? ? ? ? ? int nextc = c + acquires;

? ? ? ? ? ? ? ? if (nextc < 0)

? ? ? ? ? ? ? ? ? ? throw new Error("Maximum lock count exceeded");

? ? ? ? ? ? ? ? setState(nextc);

? ? ? ? ? ? ? ? return true;

? ? ? ? ? ? }

? ? ? ? ? ? return false;

? ? ? ? }


```

在NoFairSync中調(diào)用lock方法直接嘗試用cas的方式獲取鎖,并不去查看你是否有鎖占用,直接嘗試獲取,如果成功則直接獲取鎖,如果失敗則采用acquire的方式來獲取

而在FairSync中實現(xiàn)的tryAcquire方法中與Sync類中的noFairTryAcquire基本一致,只是在沒人獲取鎖時判斷當前隊列是否為空,或者當前線程是否在隊列的最開始位置,總得來說就是判斷是否有其他線程等待時長比當前線程要長,主要是為了保證獲取鎖的公平性

## ReenTrantLock

接下來我們看看ReenTrantLock是如何使用這兩種鎖的,首先來看看構(gòu)造方法

```java

? public ReentrantLock() {

? ? ? ? sync = new NonfairSync();

? ? }

? ? public ReentrantLock(boolean fair) {

? ? ? ? sync = fair ? new FairSync() : new NonfairSync();

? ? }

```

這里指定了鎖的公平與非公平,默認采用非公平鎖,如需公平鎖則傳入一個true即可

接下來查看lock與tryLock方法

```java

? ? public void lock() {

? ? ? ? sync.lock();

? ? }

? ? public boolean tryLock() {

? ? ? ? return sync.nonfairTryAcquire(1);

? ? }

```

如同上文所說,在lock方法中是根據(jù)當前所公平與非公平來進行而tryLock方法則是固定以非公平的方式來進行調(diào)用

以上就是ReenTrantLock的具體實現(xiàn)方式,結(jié)合AQS的兩篇文章即可將該類理解通透

![我的公眾號](https://image-xiaoazhai.oss-cn-hangzhou.aliyuncs.com/blog/qrcode_for_gh_d6d50bf01095_430.jpg)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市皱炉,隨后出現(xiàn)的幾起案子兑巾,更是在濱河造成了極大的恐慌,老刑警劉巖饰剥,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件殊霞,死亡現(xiàn)場離奇詭異,居然都是意外死亡汰蓉,警方通過查閱死者的電腦和手機绷蹲,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來顾孽,“玉大人祝钢,你說我怎么就攤上這事⊙页荩” “怎么了太颤?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長盹沈。 經(jīng)常有香客問我龄章,道長,這世上最難降的妖魔是什么乞封? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任做裙,我火速辦了婚禮,結(jié)果婚禮上肃晚,老公的妹妹穿的比我還像新娘锚贱。我一直安慰自己,他們只是感情好关串,可當我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布拧廊。 她就那樣靜靜地躺著,像睡著了一般晋修。 火紅的嫁衣襯著肌膚如雪吧碾。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天墓卦,我揣著相機與錄音倦春,去河邊找鬼。 笑死,一個胖子當著我的面吹牛睁本,可吹牛的內(nèi)容都是我干的尿庐。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼呢堰,長吁一口氣:“原來是場噩夢啊……” “哼抄瑟!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起暮胧,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤锐借,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后往衷,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體钞翔,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年席舍,在試婚紗的時候發(fā)現(xiàn)自己被綠了布轿。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡来颤,死狀恐怖汰扭,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情福铅,我是刑警寧澤萝毛,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站滑黔,受9級特大地震影響笆包,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜略荡,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一庵佣、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧汛兜,春花似錦巴粪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至漏策,卻和暖如春晶通,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背哟玷。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人巢寡。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓喉脖,卻偏偏與公主長得像,于是被迫代替她去往敵國和親抑月。 傳聞我的和親對象是個殘疾皇子树叽,可洞房花燭夜當晚...
    茶點故事閱讀 44,713評論 2 354