Java并發(fā)編程之顯式鎖機制詳解
我們之前介紹過synchronized關(guān)鍵字實現(xiàn)程序的原子性操作,它的內(nèi)部也是一種加鎖和解鎖機制,是一種聲明式的編程方式,我們只需要對方法或者代碼塊進行聲明,Java內(nèi)部幫我們在調(diào)用方法之前和結(jié)束時加鎖和解鎖。而我們本篇將要介紹的顯式鎖是一種手動式的實現(xiàn)方式,程序員控制鎖的具體實現(xiàn),雖然現(xiàn)在越來越趨向于使用synchronized直接實現(xiàn)原子操作,但是了解了Lock接口的具體實現(xiàn)機制將有助于我們對synchronized的使用。本文主要涉及以下一些內(nèi)容:
- 接口Lock的基本組成成員
- 可重入鎖ReentrantLock的基本使用
- 深入ReentrantLock的實現(xiàn)原理
一、接口Lock的基本組成成員
Lock 位于java.util.concurrent.locks包下,源碼如下:
public interface Lock {
void lock();
void lockInterruptibly()
boolean tryLock();
boolean tryLock(long time, TimeUnit unit)
void unlock();
Condition newCondition();
}
其中,
void lock();:調(diào)用該方法將獲得一個鎖的入口
lockInterruptibly():該方法也是去獲得一個鎖,但是它是響應(yīng)中斷的,一旦在獲取的過程中遭遇中斷將拋出 InterruptedException。
boolean tryLock();:該方法嘗試著去獲得一個鎖,如果獲取失敗將返回false,并不會阻塞當前線程
boolean tryLock(long time, TimeUnit unit):嘗試著去獲取一個鎖,如果獲取失敗,將阻塞等待指定的時間,期間如果能夠獲得鎖將返回true,否則返回false,響應(yīng)中斷請求。
void unlock();:釋放一個鎖
Condition newCondition();:條件變量,留待下篇文章學習
二、可重入鎖ReentrantLock的基本使用
ReentrantLock是接口 Lock的一個最主要的實現(xiàn)類,不僅實現(xiàn)了Lock中的基本的加鎖釋放鎖的方法,還擴展了自己的方法。它有兩個構(gòu)造方法:
public ReentrantLock() {
sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {
sync = fair ? new FairSync() : new NonfairSync();
}
參數(shù) fair用于保證鎖機制的公平策略,公平的策略會是的等待時間越長的線程優(yōu)先獲得鎖。保證公平必然會降低性能,所以ReentrantLock默認并不保證公平。我們用ReentrantLock來實現(xiàn)對程序的原子操作:
public class MyThread extends Thread{
private static Lock lock = new ReentrantLock();
public static int count;
@Override
public void run() {
try {
Thread.sleep((int)Math.random()*100);
lock.lock();
count++;
lock.unlock();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
當我們在主程序中啟動一百個線程隨機喚醒對count進行加一時,無論運行多少次,結(jié)果都是一百,也就是說我們的ReentrantLock是可以為我們保證原子操作的。
ReentrantLock還有一個特性就是可以重入性,即在本身獲得某個鎖的前提下可以隨意進入被該鎖鎖住的其他方法,對于一個鎖可以重復(fù)進入。除此之外,ReentrantLock還具有一些其他的有關(guān)鎖信息的方法:
- public int getHoldCount():表示當前線程持有該鎖的數(shù)量
- public boolean isHeldByCurrentThread():判斷鎖是否為當前線程持有
- public boolean isLocked():判斷鎖是否為任意一個線程持有,如果有則返回true,否則返回false
- public final boolean hasQueuedThreads():判斷該鎖上是否有線程進行等待
- public final int getQueueLength():返回當前等待隊列的長度,也就是等待進入該鎖的線程個數(shù)
三、深入ReentrantLock的實現(xiàn)原理
ReentrantLock依賴CAS和LockSupport來實現(xiàn),LockSupport有點像工具類,它主要提供兩類方法,park和unpark。
- public static void park()
- public static void parkNanos(long nanos)
- public static void parkUntil(long deadline)
- public static void unpark(Thread thread)
調(diào)用park方法會使得當前線程丟失CPU使用權(quán),從Runnable狀態(tài)轉(zhuǎn)變?yōu)閃aiting狀態(tài)。而unpark方法則反過來讓Waiting狀態(tài)的某個線程轉(zhuǎn)變狀態(tài)為Runnable,等待操作系統(tǒng)調(diào)度。parkNanos和parkUntil是和時間相關(guān)的兩個park的變種,parkNanos指定線程要等待的時間,parkUntil則指定線程要等待到什么時候,這個時間是一個絕對時間,相對于紀元的毫秒數(shù)。
Java的并發(fā)包中有很多并發(fā)工具,ReentrantReadWriteLock,Semaphore,CountDownLatch,ReentrantLock等。這些工具有很多的共同特性,于是Java為我們抽象了一個類AbstractQueuedSynchronizer(AQS)來表示這些工具的共性。ReentrantLock是其的一個實現(xiàn)類,內(nèi)部有三個內(nèi)部類:
abstract static class Sync extends AbstractQueuedSynchronizer{
//......
}
static final class NonfairSync extends Sync{
//...........
}
static final class FairSync extends Sync {
//.............
}
Sync 繼承了AQS并對其中的大部分代碼進行了簡單的實現(xiàn),F(xiàn)airSync 和NonfairSync 是針對公平策略而定義的,如果構(gòu)造ReentrantLock的時候指定公平的策略,那么其內(nèi)部的所有方法都依賴這個FairSync ,否則就全部依賴NonfairSync。接著看ReentrantLock的構(gòu)造函數(shù):
private final Sync sync;
public ReentrantLock() {
sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {
sync = fair ? new FairSync() : new NonfairSync();
}
兩個構(gòu)造方法最終會對sync進行初始化,而sync的將在后續(xù)的方法中起到相當大的作用。我們先看lock方法的具體實現(xiàn):
public void lock() {
sync.lock();
}
ReentrantLock的lock方法調(diào)用的sync的lock方法,而在sync中的lock方法是一個抽象的方法,也就是說這個方法的具體實現(xiàn)在子類中,我們看NonfairSync中的實現(xiàn):
final void lock() {
if (compareAndSetState(0, 1))
setExclusiveOwnerThread(Thread.currentThread());
else
acquire(1);
}
AQS中有一個整型類型的State變量,它用于標識當前鎖被持有的次數(shù),該值為0表示當前鎖沒有被任何線程持有。compareAndSetState是AQS中的方法,該方法調(diào)用了unsafe.compareAndSwapInt方法以CAS方式對State進行了更新,如果state的值為0,說明該鎖并沒有被任何線程持有,那么當前線程將持有該鎖并將state的值賦為1。
這就完成了獲取的動作,一旦后續(xù)的線程嘗試訪問臨界區(qū)代碼,在前面的線程沒有釋放鎖之前,將會調(diào)用 acquire(1)。
public final void acquire(int arg) {
if (!tryAcquire(arg) &&
acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
selfInterrupt();
}
tryAcquire還是調(diào)用了AQS中的實現(xiàn),
final boolean nonfairTryAcquire(int acquires) {
final Thread current = Thread.currentThread();
int c = getState();
if (c == 0) {
if (compareAndSetState(0, acquires)) {
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;
}
第一個if判斷,想要持有的鎖是否被持有(雖然之前判斷過了,但是有可能在我們調(diào)用nonfairTryAcquire方法的期間,之前的線程釋放了該鎖),如果未被任何線程持有,那么將直接持有該鎖。
第二個if判斷,如果當前鎖的持有者就是當前線程,表示這是同線程的重入操作,于是增加鎖定次數(shù)并設(shè)置state的值。
整個方法結(jié)束之后,如果當前線程獲得了鎖,都將返回true,否則都會返回false。而如果tryAcquire方法返回true,那么整個acquire方法也將結(jié)束,否則就說明當前線程并沒有通過鎖,需要被阻塞。那么就會調(diào)用acquireQueued(addWaiter(Node.EXCLUSIVE), arg)方法。
private Node addWaiter(Node mode) {
Node node = new Node(Thread.currentThread(), mode);
Node pred = tail;
if (pred != null) {
node.prev = pred;
if (compareAndSetTail(pred, node)) {
pred.next = node;
return node;
}
}
enq(node);
return node;
}
addWaiter方法將當前線程包裹成一個Node結(jié)點,添加到AQS內(nèi)部所維護的一個等待隊列并返回該Node結(jié)點。最后調(diào)用acquireQueued方法:
final boolean acquireQueued(final Node node, int arg) {
boolean failed = true;
try {
boolean interrupted = false;
for (;;) {
final Node p = node.predecessor();
if (p == head && tryAcquire(arg)) {
setHead(node);
p.next = null; // help GC
failed = false;
return interrupted;
}
if (shouldParkAfterFailedAcquire(p, node) &&
parkAndCheckInterrupt())
interrupted = true;
}
} finally {
if (failed)
cancelAcquire(node);
}
}
該方法首先會去獲得node的前一個結(jié)點,判斷如果是head結(jié)點,那么說明當前的node結(jié)點是整個等待隊列上的第一個等待的結(jié)點。于是讓它嘗試著去獲得鎖,如果能夠獲得鎖,將從等待隊列中清除它并返回。
如果發(fā)現(xiàn)當前結(jié)點前面還有等待的結(jié)點或者嘗試獲取鎖失敗,那么將會調(diào)用shouldParkAfterFailedAcquire方法判斷該結(jié)點鎖對應(yīng)的線程是否需要被unpark阻塞,并最終調(diào)用LockSupport.park(this)阻塞當前線程。
在第一個線程持有該鎖的前提下,成功阻塞了第二個線程。這大概就是整個lock方法的調(diào)用鏈流程。
接下來看看unlock的具體實現(xiàn),
public void unlock() {
sync.release(1);
}
這是ReentrantLock中對AQS的unlock的具體實現(xiàn),調(diào)用了sync的release方法,這個方法是其父類AQS中的方法:
public final boolean release(int arg) {
if (tryRelease(arg)) {
Node h = head;
if (h != null && h.waitStatus != 0)
unparkSuccessor(h);
return true;
}
return false;
}
tryRelease被sync重寫,具體代碼如下:
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;
}
首先判斷如果當前線程并不是鎖的當前持有者,拋出異常(不持有該鎖自然不能釋放該鎖)。如果c等于0則表示,當前鎖只被持有一次,也就是當前線程并沒有多次重入該鎖,于是將該鎖的持有者設(shè)置為null,表示未被任何線程持有。如果c不等于0,那么說明該鎖被當前線程重入多次,于是對state減一并設(shè)置state的值。最終如果返回true則說明該鎖被釋放了,否則說明當前線程依然持有該鎖。
回到release方法,如果tryRelease(arg)返回true,那么方法體會判斷當前等待隊列是否有結(jié)點在等待該鎖,如果有則調(diào)用unparkSuccessor(h)方法喚醒等待隊列上的第一個等待的結(jié)點線程并返回true。
這里有一個細節(jié),其實所有未能獲得鎖的線程都被阻塞在方法中:
final boolean acquireQueued(final Node node, int arg) {
boolean failed = true;
try {
boolean interrupted = false;
for (;;) {
final Node p = node.predecessor();
if (p == head && tryAcquire(arg)) {
setHead(node);
p.next = null; // help GC
failed = false;
return interrupted;
}
if (shouldParkAfterFailedAcquire(p, node) &&
parkAndCheckInterrupt())
//******等待線程喚醒的起始位置********//
interrupted = true;
}
} finally {
if (failed)
cancelAcquire(node);
}
}
未能獲得鎖的線程被方法parkAndCheckInterrupt阻塞了,所以當我們在unlock中調(diào)用unpark喚醒一個等待隊列上的線程結(jié)點時,線程將從此處重新進入死循環(huán)嘗試去獲取鎖。如果能夠獲得鎖,將從等待隊列中移除自己,并返回,否則再次被阻塞等待喚醒。
整個unlock方法的執(zhí)行流程也已經(jīng)大致介紹完成,最后我們看看可重入鎖ReentrantLock和synchronized的一些對比。
四、ReentrantLock對比synchronized
synchronized更傾向于一種聲明式的編程方式,我們在方法前使用synchronized修飾,Java會自動為我們實現(xiàn)其內(nèi)部的細節(jié),什么時候加鎖,什么時候釋放鎖都是它負責的。
而對于我們的ReentrantLock重入鎖來說,需要我們自己手動的去加鎖和釋放鎖,對于邏輯的要求更高,也相對更難。
而隨著jvm版本的更新和優(yōu)化,ReentrantLock和synchronized在性能上的差別在逐漸縮小,所以一般建議使用synchronized而盡量避免復(fù)雜難操作的ReentrantLock。
對于顯式鎖的基本情況大致介紹如上,如有錯誤之處,望指出!
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot用實體接收Get請求傳遞過來的多個參數(shù)的兩種方式
本文主要介紹SpringBoot用實體接收Get請求傳遞過來的多個參數(shù),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-04-04
SpringBoot整合Hibernate Validator實現(xiàn)參數(shù)驗證功能
這篇文章主要介紹了SpringBoot整合Hibernate Validator實現(xiàn)參數(shù)驗證功能,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-06-06
Java的wait(), notify()和notifyAll()使用心得
本篇文章是對java的 wait(),notify(),notifyAll()進行了詳細的分析介紹,需要的朋友參考下2013-08-08
Springboot?+redis+谷歌開源Kaptcha實現(xiàn)圖片驗證碼功能
這篇文章主要介紹了Springboot?+redis+?歌開源Kaptcha實現(xiàn)圖片驗證碼功能,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-01-01

