Java併發編程鎖系列之ReentrantLock對象總結

Java併發編程鎖系列之ReentrantLock對象總結

在Java併發編程中,根據不同維度來區分鎖的話,鎖可以分為十五種。ReentranckLock就是其中的多個分類。

本文主要內容:重入鎖理解;重入鎖代碼演示; ReentranckLock的總結

本篇是《凱哥(凱哥Java:kagejava)併發編程學習》系列之《Lock系列》教程的第四篇:《Java併發包下鎖學習第六篇:ReentranckLock的總結》。

我們先來看看內部結構:

Java併發編程鎖系列之ReentrantLock對象總結

ReentranckLock內部有三個內部類,分別是:

Sync:繼承了AbstractQueuedSynchronizer(AQS)同步器的內部類,來實現同步機制的;

FairSync:公平鎖對象;

NonfairSync:非公平鎖對象。

關於公平鎖與非公平鎖詳細介紹文章見:《Java併發編程鎖之獨佔公平鎖與非公平鎖比較》和《Java並非鎖之獨佔非公平鎖理解》兩篇文章。

再來看看對象名稱:Reentranck的中文意思:再入、重入的意思。即該對象還是重入鎖。

公平鎖和非公平鎖在獲取鎖的區別在於:

公平鎖獲取鎖的時候,進入排隊。源碼如下圖:

Java併發編程鎖系列之ReentrantLock對象總結

非公平鎖線嘗試插隊,如果插隊不成功再進行排隊。源碼如下圖:

Java併發編程鎖系列之ReentrantLock對象總結

那麼什麼是重入鎖呢?

重入鎖(遞歸鎖)可以理解為:同一個線程函數獲得鎖之後,內層遞歸函數依然能夠獲取到該鎖對象的代碼,也即,在同一個線程的外層方法訪問的時候,獲取到了鎖,在進入內層方法後能夠自動獲取到鎖。線程可以進入任何一個它已經擁有的鎖所同步著的代碼塊。額,說的啥意思?每個中文都認識,但是組合在一起,就不知道啥意思了。

我們來舉個生活中的例子:

在現實生活中,我們一般只需要帶有自己大門的鑰匙(當然,如果是合租的朋友還需要帶著自己房間的鑰匙)。當我們開了大門的鑰匙,進入房間後,我們在去廚房或者是去衛生間的時候,不用在拿鑰匙開廚房或者衛生間的門了吧。為啥呢?因為我們已經已經有大門的鎖的鑰匙並且已經進入到了房間了。廚房和衛生間已經在大門鎖管理的範圍內了。這種場景站在併發鎖的角度來看的話:一同一個線程函數獲得鎖之後(你拿著鑰匙打開了大門之後),內層遞歸函數依然能夠獲取到該鎖對象的代碼(進入房間後,房間內的廚房衛生間可以隨便出入)。這樣是不是就好理解了?

如果還是不理解的話,可以想想spring默認的事務傳遞方式。一個方法開啟了事務,默認這個方法調用其他方法,也會使用這個事務。這樣是不是就好理解了。

重入鎖代碼演示

先來看看,被線程操作的資源類:message對象

Java併發編程鎖系列之ReentrantLock對象總結


在message對象中,sendMsg使用了鎖,sendEmail方法也使用了鎖,而且在sendMesg方法中調用了sendEmail方法。

這個時候使用三個線程來操作:

Java併發編程鎖系列之ReentrantLock對象總結


運行結果:

Java併發編程鎖系列之ReentrantLock對象總結


我們從運行的結果中,可以看出,同一個線程操作了兩個帶有鎖的方法。

說明ReentrantLock是可重入鎖。

總結:

經過前面幾篇及本篇的學習,我們可以對ReentrantLock(以下簡稱:RLock)得到如下總結

1:從線程是否需要對資源進行加鎖方面來看的話,RLock是悲觀鎖;

2:從鎖是否需要排隊的公平性來區分的話,RLock的FairSync對象是公平鎖而NonfairSync對象是非公平鎖;

3:從多個線程能共享一把鎖的角度來分的話RLock屬於排他鎖(獨佔式鎖);

從鎖是否重複獲取角度來看,屬於可重入鎖


分享到:


相關文章: