<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
摘要: 在Java中提供了synchronized關鍵字來保證只有一個執行緒能夠存取同步程式碼塊。既然已經提供了synchronized關鍵字,那為何在Java的SDK包中,還會提供Lock介面呢?這是不是重複造輪子,多此一舉呢?今天,我們就一起來探討下這個問題。
在Java中提供了synchronized
關鍵字來保證只有一個執行緒能夠存取同步程式碼塊。既然已經提供了synchronized
關鍵字,那為何在Java的SDK包中,還會提供Lock介面呢?這是不是重複造輪子,多此一舉呢?今天,我們就一起來探討下這個問題。
問題?
既然JVM中提供了synchronized關鍵字來保證只有一個執行緒能夠存取同步程式碼塊,為何還要提供Lock介面呢?這是在重複造輪子嗎?Java的設計者們為何要這樣做呢?讓我們一起帶著疑問往下看。
很多小夥伴可能會聽說過,在Java 1.5版本中,synchronized的效能不如Lock,但在Java 1.6版本之後,synchronized
做了很多優化,效能提升了不少。那既然synchronized關鍵字的效能已經提升了,那為何還要使用Lock呢?
如果我們向更深層次思考的話,就不難想到了:我們使用synchronized
加鎖是無法主動釋放鎖的,這就會涉及到死鎖的問題。
如果要發生死鎖,則必須存在以下四個必要條件,四者缺一不可。
在一段時間內某資源僅為一個執行緒所佔有。此時若有其他執行緒請求該資源,則請求執行緒只能等待。
執行緒所獲得的資源在未使用完畢之前,不能被其他執行緒強行奪走,即只能由獲得該資源的執行緒自己來釋放(只能是主動釋放)。
執行緒已經保持了至少一個資源,但又提出了新的資源請求,而該資源已被其他執行緒佔有,此時請求執行緒被阻塞,但對自己已獲得的資源保持不放。
在發生死鎖時必然存在一個程序等待佇列{P1,P2,…,Pn},其中P1等待P2佔有的資源,P2等待P3佔有的資源,…,Pn等待P1佔有的資源,形成一個程序等待環路,環路中每一個程序所佔有的資源同時被另一個申請,也就是前一個程序佔有後一個程序所深情地資源。
如果我們的程式使用synchronized
關鍵字發生了死鎖時,synchronized關鍵是是無法破壞“不可剝奪”這個死鎖的條件的。這是因為synchronized申請資源的時候, 如果申請不到, 執行緒直接進入阻塞狀態了, 而執行緒進入阻塞狀態, 啥都幹不了, 也釋放不了執行緒已經佔有的資源。
然而,在大部分場景下,我們都是希望“不可剝奪”這個條件能夠被破壞。也就是說對於“不可剝奪”這個條件,佔用部分資源的執行緒進一步申請其他資源時, 如果申請不到, 可以主動釋放它佔有的資源, 這樣不可剝奪這個條件就破壞掉了。
如果我們自己重新設計鎖來解決synchronized
的問題,我們該如何設計呢?
瞭解了synchronized的侷限性之後,如果是讓我們自己實現一把同步鎖,我們該如何設計呢?也就是說,我們在設計鎖的時候,要如何解決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();
支援中斷。
tryLock()方法是有返回值的,它表示用來嘗試獲取鎖,如果獲取成功,則返回true,如果獲取失敗(即鎖已被其他執行緒獲取),則返回false,也就說這個方法無論如何都會立即返回。在拿不到鎖時不會一直在那等待。
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中提供synchronized後為什麼還要提供Lock的文章就介紹到這了,更多相關Java中 synchronized Lock內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45