Java并發教程–重入鎖

Java的synced關鍵字是一個很棒的工具–它使我們能夠以一種簡單可靠的方式來同步對關鍵部分的訪問,而且也不難理解。

但是有時我們需要對同步進行更多控制。 我們要么需要分別控制訪問類型(讀取和寫入),要么使用起來很麻煩,因為要么沒有明顯的互斥鎖,要么我們需要維護多個互斥鎖。

值得慶幸的是,Java 1.5中添加了鎖實用程序類,使這些問題更易于解決。

Java重入鎖

Java在java.util.concurrent.locks包中有一些鎖實現。

鎖的一般類很好地布置為接口:

  • –最簡單的鎖,可以獲取和釋放
  • ReadWriteLock –具有讀和寫鎖類型的鎖實現–一次可以持有多個讀鎖,除非持有排他寫鎖

Java提供了我們關心的這些鎖的兩種實現–兩者都是可重入的(這僅意味著線程可以多次重新獲取同一鎖而沒有任何問題)。

  • ReentrantLock –如您所料,可重入鎖實現
  • ReentrantReadWriteLock –可重入ReadWriteLock實現

現在,讓我們看一些例子。

讀/寫鎖示例

那么如何使用鎖呢? 這很簡單:只需獲取并發布(永遠不要忘記發布-終于是您的朋友!)。

假設我們有一個非常簡單的情況,我們需要同步訪問一對變量。 一個是簡單的值,另一個是根據一些冗長的計算得出的。 首先,這就是我們如何使用synced關鍵字執行此操作。

public class Calculator {private int calculatedValue;private int value;public synchronized void calculate(int value) {this.value = value;this.calculatedValue = doMySlowCalculation(value);}public synchronized int getCalculatedValue() {return calculatedValue;}public synchronized int getValue() {return value;}
}

很簡單,但是如果我們有很多爭用或者執行大量讀取而寫入很少,則同步可能會影響性能。 由于頻繁讀取比寫入頻繁得多,因此使用ReadWriteLock可幫助我們最大程度地減少問題:

public class Calculator {private int calculatedValue;private int value;private ReadWriteLock lock = new ReentrantReadWriteLock();public void calculate(int value) {lock.writeLock().lock();try {this.value = value;this.calculatedValue = doMySlowCalculation(value);} finally {lock.writeLock().unlock();}}public int getCalculatedValue() {lock.readLock().lock();try {return calculatedValue;} finally {lock.readLock().unlock();}}public int getValue() {lock.readLock().lock();try {return value;} finally {lock.readLock().unlock();}}
}

該示例實際上顯示了使用同步的has的一個大優點:與使用顯式鎖相比,此方法簡潔明了且更加安全。 但是鎖提供了使用靈活性,而這是我們以前所沒有的。

在上面的示例中,我們可以讓數百個線程一次讀取相同的值而不會出現問題,并且只有在獲得寫入鎖定時才阻塞讀取器。 請記住:許多讀取器可以同時獲取讀取鎖定,但是在獲取寫入鎖定時不允許讀取器或寫入器。

更典型的用途

我們的第一個示例可能會讓您感到困惑或不完全相信顯式鎖是有用的。 難道他們還沒有其他用途嗎? 當然!

我們在Carfey使用顯式鎖來解決許多問題。 一個示例是您有可以同時運行的各種任務,但是您不希望同時運行多個相同類型的任務。 一種實現它的干凈方法是使用鎖。 可以通過同步來完成,但是鎖使我們能夠在超時后失敗。

值得一提的是,您會注意到我們使用了同步鎖和顯式鎖的組合-有時一個比另一個更干凈,更簡單。

public class TaskRunner {private Map<Class<? extends Runnable>,  Lock> mLocks =new HashMap<Class<? extends Runnable>,  Lock>();public void runTaskUniquely(Runnable r, int secondsToWait) {Lock lock = getLock(r.getClass());boolean acquired = lock.tryLock(secondsToWait, TimeUnit.SECONDS);if (acquired) {try {r.run();} finally {lock.unlock();}} else {// failure code here}}private synchronized Lock getLock(Class clazz) {Lock l = mLocks.get(clazz);if (l == null) {l = new ReentrantLock();mLocks.put(clazz, l);}return l;}
}

這兩個示例應該使您對如何同時使用計劃ReadWriteLocks有所了解。 與同步一樣,不必擔心重新獲得相同的鎖-JDK中提供的鎖是可重入的,因此不會有任何問題。

每當您處理并發時,都有危險。 永遠記住以下幾點:

  • 釋放finally塊中的所有鎖。 這是規則1,有一個原因。
  • 當心線程饑餓! 如果您有不想永久等待的許多讀者和偶爾的作家,那么ReentrantLocks中的公平設置可能會很有用。 如果其他線程不斷持有讀取鎖,那么編寫者可能會等待很長時間(可能永遠)。
  • 盡可能使用同步。 您將避免錯誤并保持代碼清潔。
  • 如果您不希望線程無限期等待獲取鎖,請使用tryLock() -這類似于數據庫具有的等待鎖超時。

就是這樣! 如果您有任何問題或意見,請隨時將其留在下面。

參考: Java并發第2部分–來自我們的JCG合作伙伴的Carent博客上的 Reentrant Locks 。

相關文章 :
  • Java并發教程–信號量
  • Java并發教程–線程池
  • Java并發教程–可調用,將來
  • Java并發教程–阻塞隊列
  • Java并發教程– CountDownLatch
  • Exchanger和無GC的Java
  • Java Fork / Join進行并行編程
  • Java最佳實踐–隊列之戰和鏈接的ConcurrentHashMap
  • 使用迭代器時如何避免ConcurrentModificationException
  • 改善Java應用程序性能的快速技巧

翻譯自: https://www.javacodegeeks.com/2011/09/java-concurrency-tutorial-reentrant.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/374122.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/374122.shtml
英文地址,請注明出處:http://en.pswp.cn/news/374122.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

找出互聯網類似以下圖的實例

轉載于:https://www.cnblogs.com/sghcjy/p/4978851.html

python比較運算符重載_python運算符重載

1、打印操作會首先嘗試__str__和str內置函數&#xff0c;他通常返回一個用戶友好顯示。__repr__用于所有其他環境&#xff0c;用于交互式模式下提示回應以及repr函數&#xff0c;如果沒有使用__str__&#xff0c;則會使用print和str。它通常返回一個編碼字符串&#xff0c;可以…

使用Spring MVC開發Restful Web服務

REST簡介 摘自Wikipedia&#xff1a; REST風格的體系結構由客戶端和服務器組成。 客戶端向服務器發起請求&#xff1b; 服務器處理請求并返回適當的響應。 請求和響應圍繞資源表示的傳遞而構建。 資源本質上可以是可以解決的任何連貫且有意義的概念。 正如您所閱讀的&#xff0…

深入Java核心 Java內存分配原理精講

深入Java核心 Java內存分配原理精講 Java內存分配與管理是Java的核心技術之一&#xff0c;之前我們曾介紹過Java的內存管理與內存泄露以及Java垃圾回收方面的知識&#xff0c;今天我們再次深入Java核心&#xff0c;詳細介紹一下Java在內存分配方面的知識。一般Java在內存分配時…

iOS正則表達式(親測,持續更新)

先來說說判斷方法,書寫不簡介但是好理解: -(BOOL)isRealNmaeString:(NSString *)str{NSString *pattern "填寫正則表達式";NSPredicate *pred [NSPredicate predicateWithFormat:"SELF MATCHES %", pattern];BOOL isMatch [pred evaluateWithObject:str…

python新建一個文件夾需要重新安裝模塊嗎_解決pycharm每次新建項目都要重新安裝一些第三方庫的問題...

目前有三個解決辦法&#xff0c;也是親測有用的&#xff1a;第一個方法&#xff1a;因為之前有通過pycharm的project interpreter里的號添加過一些庫&#xff0c;但添加的庫只是指定的項目用的&#xff0c;如果想要用&#xff0c;就必須用之前的項目的python解釋器&#xff0c;…

端到端測試的濫用–測試技術2

我的上一個博客是有關測試代碼方法的一系列博客中的第一篇&#xff0c;概述了使用一種非常常見的模式從數據庫檢索地址的簡單方案&#xff1a; …并描述了一種非常通用的測試技術&#xff1a; 不編寫測試 &#xff0c; 而是手動進行所有操作。 今天的博客涵蓋了另一種實踐&…

[AlwaysOn Availability Groups]排查:AG超過RPO

[AlwaysOn Availability Groups]排查&#xff1a;AG超過RPO 排查&#xff1a;AG超過RPO 在異步提交的secondary上執行了切換&#xff0c;你可能會發現數據的丟失大于RPO&#xff0c;或者在計算可以忍受的數據都是超過了RPO。 1.通常原因 1.網絡延遲太高&#xff0c;網絡吞吐量太…

那些年困擾我們的Linux 的蠕蟲、病毒和木馬

雖然針對Linux的惡意軟件并不像針對Windows乃至OS X那樣普遍&#xff0c;但是近些年來&#xff0c;Linux面臨的安全威脅卻變得越來越多、越來越嚴重。個中原因包括&#xff0c;手機爆炸性的普及意味著基于Linux的安卓成為惡意黑 客最具吸引力的目標之一&#xff0c;以及使用Lin…

python單元測試框架unittest介紹和使用_Python+Selenium框架設計篇之-簡單介紹unittest單元測試框架...

前面文章已經簡單介紹了一些關于自動化測試框架的介紹&#xff0c;知道了什么是自動化測試框架&#xff0c;主要有哪些特點&#xff0c;基本組成部分等。在繼續介紹框架設計之前&#xff0c;我們先來學習一個工具&#xff0c;叫unittest。unittest是一個單元測試框架&#xff0…

使用PowerMock模擬靜態方法

在最近的博客中&#xff0c;我試圖強調使用依賴注入的好處&#xff0c;并表達一種想法&#xff0c;即這種技術的主要好處之一是&#xff0c;通過在類之間提供高度的隔離&#xff0c;它可以使您更輕松地測試代碼&#xff0c;并且得出的結論是&#xff0c;許多好的測試等于好的代…

多態之向上轉型

//向上轉型&#xff0c;子類引用指向父類對象 public class UpcastingDemo{ public static void main(String[] args){ Employee enew Employee(); System.out.println(e.grade); e.job(); e.run(); System.out.println("\n"); Manager mnew Manager(…

(轉)FPGA異步時序和多時鐘模塊

http://bbs.ednchina.com/BLOG_ARTICLE_3019907.HTM 第六章 時鐘域 有一個有趣的現象&#xff0c;眾多數字設計特別是與FPGA設計相關的教科書都特別強調整個設計最好采用唯一的時鐘域。換句話說&#xff0c;只有一個獨立的網絡可以驅動一個設計中所有觸發器的時鐘端口。雖然…

穆里尼奧:與范加爾風格不同,轉變需要時間

據英媒報道&#xff0c;曼聯主帥穆里尼奧近日向媒體表示自己很難繼續遵循前任主帥范加爾的理念去建立球隊&#xff0c;因為他們兩人有著完全不同的想法。 穆里尼奧近日在接受BT Sport的采訪時表示&#xff1a;“這份工作對于我來說最難的地方便是我與范加爾是非常不同的教練&am…

怎么檢測不到我的音頻_Linux 上的最佳音頻編輯工具推薦 | Linux 中國

在 Linux 上&#xff0c;有很多種音頻編輯器可供你選用。不論你是一個專業的音樂制作人&#xff0c;還是只想學學怎么做出超棒的音樂的愛好者&#xff0c;這些強大的音頻編輯器都是很有用的工具。-- Ankush Das(作者)在 Linux 上&#xff0c;有很多種音頻編輯器可供你選用。不論…

具有GlassFish和一致性的高性能JPA –第3部分

在我的四部分系列的第三部分中&#xff0c;我將解釋將Coherence與EclipseLink和GlassFish結合使用的第二種策略。 這就是通過EclipseLink使用Coherence作為二級緩存&#xff08;L2&#xff09;的全部內容。 一般的做法 這種方法將Coherence數據網格應用于依賴于無法完全預加載到…

接口使用時注意

interface Service{ void doSome(); //方法的默認修飾符為public abstract } public class InterfaceNote implements Service{ //方法默認的修飾符為 default void doSome(){ System.out.println("做一些服務&#xff01;"); } public static void main(String…

【Hexo】deploy出錯的解決方法

1、ERROR Deployer not found: git 執行npm install hexo-deployer-git --save 2、$ hexo d INFO Deploying: git INFO Clearing .deploy_git folder... INFO Copying files from public folder... fatal: Not a git repository (or any of the parent directories): .git F…

精選30道Java筆試題解答(轉)

都是一些非常非常基礎的題&#xff0c;是我最近參加各大IT公司筆試后靠記憶記下來的&#xff0c;經過整理獻給與我一樣參加各大IT校園招聘的同學們&#xff0c;純考Java基礎功底&#xff0c; 老手們就不用進來了&#xff0c;免得笑話我們這些未出校門的孩紙們&#xff0c;但是I…

list取值_Redis中List及quicklist實現-2

上一篇中看了List的使用方式、quicklist中的各個結構體&#xff0c;這一篇來看看quicklist里面的幾個核心函數&#xff0c;quicklistCreate函數、quicklistCreateNode函數、quicklistPush函數、quicklistPop函數。接下來我們通過源碼看一下quicklist中是如何借鑒STL中deque的這…