首頁> Java> java教程> 主體

Java中為什麼需要提供Lock,而不只使用synchronized關鍵字?

PHPz
發布: 2023-04-20 17:01:17
轉載
1382 人瀏覽過

摘要:在Java中提供了synchronized關鍵字來保證只有一個執行緒能夠存取同步程式碼區塊。既然已經提供了synchronized關鍵字,為何在Java的SDK包中,還會提供Lock介面呢?這是不是重複造輪子,多此一舉呢?今天,我們就一起來探討下這個問題。

在Java中提供了synchronized關鍵字來保證只有一個執行緒能夠存取同步程式碼區塊。既然已經提供了synchronized關鍵字,那為何在Java的SDK套件中,還會提供Lock介面呢?這是不是重複造輪子,多此一舉呢?今天,我們就一起來探討下這個問題。

問題?

既然JVM中提供了synchronized關鍵字來保證只有一個執行緒能夠存取同步程式碼區塊,為何還要提供Lock介面呢?這是在重複造輪子嗎? Java的設計者為何要這樣做呢?讓我們一起帶著疑問往下看。

一、為何提供Lock介面?

很多小夥伴可能會聽說過,在Java 1.5版本中,synchronized的效能不如Lock,但在Java 1.6版本之後,synchronized做了很多優化,效能提升了不少。那既然synchronized關鍵字的效能已經提升了,那為何還要使用Lock呢?

如果我們向更深層次思考的話,就不難想到了:我們使用synchronized加鎖是無法主動釋放鎖的,這就會涉及到死鎖的問題。

二、死鎖問題

如果要發生死鎖,則必須存在以下四個必要條件,四者缺一不可。

Java中為什麼需要提供Lock,而不只使用synchronized關鍵字?

  • #互斥條件

在一段時間內某資源僅為一個執行緒所佔有。此時若有其他執行緒請求該資源,則請求執行緒只能等待。

  • 不可剝奪條件

#執行緒所獲得的資源在未使用完畢之前,不能被其他線程強行奪走,即只能由獲得該資源的線程自己來釋放(只能是主動釋放)。

  • 請求與保持條件

#執行緒已經保持了至少一個資源,但又提出了新的資源請求,而該資源已被其他執行緒佔有,此時請求執行緒被阻塞,但對自己已獲得的資源保持不放。

  • 循環等待條件

#在發生死鎖時必然存在一個行程等待佇列{P1,P2,&hellip ;,Pn},其中P1等待P2佔有的資源,P2等待P3佔有的資源,…,Pn等待P1佔有的資源,形成一個進程等待環路,環路中每一個進程所佔有的資源同時被另一個申請,也就是前一個進程佔有後一個進程所深情地資源。

三、synchronized的限制

如果我們的程式使用synchronized關鍵字發生了死鎖時,synchronized關鍵是無法破壞「不可剝奪」這個死鎖的條件的。這是因為synchronized申請資源的時候, 如果申請不到, 線程直接進入阻塞狀態了, 而線程進入阻塞狀態, 啥都乾不了, 也釋放不了線程已經佔有的資源。

然而,在大部分場景下,我們都是希望「不可剝奪」這個條件能夠被破壞。也就是說對於「不可剝奪」這個條件,佔用部分資源的線程進一步申請其他資源時, 如果申請不到, 可以主動釋放它佔有的資源, 這樣不可剝奪這個條件就破壞掉了。

如果我們自己重新設計鎖定來解決synchronized的問題,我們該如何設計呢?

四、解決問題

了解了synchronized的限制之後,如果是讓我們自己實作一個同步鎖,我們該如何設計呢?也就是說,我們在設計鎖的時候,要如何解決synchronized的限制問題呢?這裡,我覺得可以從三個面向來思考這個問題。

Java中為什麼需要提供Lock,而不只使用synchronized關鍵字?

(1)能夠回應中斷。synchronized的問題是, 持有鎖定A後, 如果嘗試取得鎖定B失敗, 那麼執行緒就進入阻塞狀態, 一旦發生死鎖, 就沒有任何機會來喚醒阻塞的執行緒。但如果阻塞狀態的執行緒能夠回應中斷訊號, 也就是說當我們給阻塞的執行緒發送中斷訊號的時候, 能夠喚醒它, 那它就有機會釋放曾經持有的鎖定A。這樣就破壞了不可剝奪條件了。

(2)支援逾時。如果執行緒在一段時間之內沒有取得到鎖, 不是進入阻塞狀態, 而是回傳一個錯誤, 那這個執行緒也有機會釋放曾經持有的鎖。這樣也能破壞不可剝奪條件。

(3)非阻塞地取得鎖。如果嘗試取得鎖定失敗, 並不進入阻塞狀態, 而是直接返回, 那這個線程也有機會釋放曾經持有的鎖。這樣也能破壞不可剝奪條件。

體現在Lock介面上,就是Lock介面提供的三個方法,

如下:

// 支持中断的API void lockInterruptibly() throws InterruptedException; // 支持超时的API boolean tryLock(long time, TimeUnit unit) throws InterruptedException; // 支持非阻塞获取锁的API boolean tryLock();
登入後複製
  • lockInterruptibly()

支援中斷。

  • tryLock()方法

#tryLock()方法是有回傳值的,它表示用來嘗試取得鎖,如果取得成功,則回傳true,如果取得失敗(即鎖已被其他執行緒取得),則傳回false,也就說這個方法無論如何都會立即回傳。在拿不到鎖時不會一直在那裡等待。

  • tryLock(long time, TimeUnit unit)方法

tryLock(long time, TimeUnit unit)方法和tryLock ()方法是類似的,只不過差別在於這個方法在拿不到鎖時會等待一定的時間,在時間期限之內如果還拿不到鎖,就回傳false。如果一開始拿到鎖或在等待期間內拿到了鎖,則回傳true。

也就是說,對於死鎖問題,Lock能夠破壞不可剝奪的條件,例如,我們下面的程式碼就破壞了死鎖的不可剝奪的條件。

public class TansferAccount{ private Lock thisLock = new ReentrantLock(); private Lock targetLock = new ReentrantLock(); //账户的余额 private Integer balance; //转账操作 public void transfer(TansferAccount target, Integer transferMoney){ boolean isThisLock = thisLock.tryLock(); if(isThisLock){ try{ boolean isTargetLock = targetLock.tryLock(); if(isTargetLock){ try{ if(this.balance >= transferMoney){ this.balance -= transferMoney; target.balance += transferMoney; } }finally{ targetLock.unlock } } }finally{ thisLock.unlock(); } } } }
登入後複製

例外,Lock下面有一個ReentrantLock,而ReentrantLock支援公平鎖和非公平鎖。

在使用ReentrantLock的時候, ReentrantLock中有兩個建構函數, 一個是無參建構函數, 一個是傳入fair參數的建構子。 fair參數代表的是鎖的公平策略, 如果傳入true就表示需要建構一個公平鎖, 反之則表示要建構一個非公平鎖。如下程式碼片段所示。

//无参构造函数: 默认非公平锁 public ReentrantLock() { sync = new NonfairSync(); } //根据公平策略参数创建锁 public ReentrantLock(boolean fair){ sync = fair ? new FairSync() : new NonfairSync(); }
登入後複製

鎖定的實作在本質上都對應著一個入口等待佇列, 如果一個執行緒沒有取得鎖, 就會進入等待佇列, 當有執行緒釋放鎖的時候, 就需要從等待佇列中喚醒一個等待的線程。如果是公平鎖, 喚醒的策略是誰等待的時間長, 就喚醒誰, 很公平; 如果是非公平鎖, 則不提供這個公平保證, 有可能等待時間短的線程反而先被喚醒。而Lock是支持公平鎖的,synchronized不支援公平鎖。

最後,值得注意的是,在使用Lock加鎖時,一定要在finally{}程式碼區塊中釋放鎖,例如,下面的程式碼片段所示。

try{ lock.lock(); }finally{ lock.unlock(); }
登入後複製

註:其他synchronized和Lock的詳細說明,小夥伴們自行查閱即可。

以上是Java中為什麼需要提供Lock,而不只使用synchronized關鍵字?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:yisu.com
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板
關於我們 免責聲明 Sitemap
PHP中文網:公益線上PHP培訓,幫助PHP學習者快速成長!