java并發(fā)編程中ReentrantLock可重入讀寫(xiě)鎖
一、ReentrantLock可重入鎖
可重入鎖ReentrantLock 是一個(gè)互斥鎖,即同一時(shí)間只有一個(gè)線(xiàn)程能夠獲取鎖定資源,執(zhí)行鎖定范圍內(nèi)的代碼。這一點(diǎn)與synchronized 關(guān)鍵字十分相似。其基本用法代碼如下:
Lock lock = new ReentrantLock(); //實(shí)例化鎖
//lock.lock(); //上鎖
boolean locked = lock.tryLock(); //嘗試上鎖
if(locked){
try {
//被鎖定的同步代碼塊,同時(shí)只能被一個(gè)線(xiàn)程執(zhí)行
}finally {
lock.unlock(); //放在finally代碼塊中,保證鎖一定會(huì)被釋放
}
}
通過(guò)lock函數(shù)獲取鎖,通過(guò)unlock函數(shù)釋放鎖。非常重要的是,需要把需要同步執(zhí)行的代碼放入 try/finally 代碼塊中,并在finally中將鎖釋放。ReentrantLock是可重入鎖,即:(lock/unlok)動(dòng)作里面可以嵌套(lock/unlock),針對(duì)同一個(gè)鎖可以多次嵌套使用,不會(huì)產(chǎn)生死鎖。但是lock函數(shù)與unlock函數(shù)在代碼中必須成對(duì)出現(xiàn),否則會(huì)出現(xiàn)死鎖。
二、ReentrantReadWriteLock讀寫(xiě)鎖
ReentrantReadWriteLock類(lèi)為讀寫(xiě)鎖實(shí)現(xiàn)類(lèi),針對(duì)某一個(gè)對(duì)象或可變變量,只要沒(méi)有線(xiàn)程在修改它,這個(gè)對(duì)象或可變變量就可以同時(shí)被多個(gè)線(xiàn)程讀取。ReentrantReadWriteLock將鎖分為讀鎖和寫(xiě)鎖,只要沒(méi)有線(xiàn)程持有寫(xiě)鎖的情況下,讀鎖可以由多個(gè)線(xiàn)程同時(shí)持有。
- 讀鎖-如果沒(méi)有線(xiàn)程獲取或請(qǐng)求寫(xiě)鎖,那么多個(gè)線(xiàn)程可以獲取讀鎖
- 寫(xiě)鎖-如果沒(méi)有線(xiàn)程在讀或?qū)?,那么只有一個(gè)線(xiàn)程可以獲得寫(xiě)鎖
簡(jiǎn)單的說(shuō)就是ReentrantReadWriteLock可以保證最多同時(shí)有一個(gè)線(xiàn)程在寫(xiě)數(shù)據(jù),或者可以同時(shí)有多個(gè)線(xiàn)程讀數(shù)據(jù)。因此使用ReentrantReadWriteLock,在讀操作比寫(xiě)操作更頻繁的情況下,可以提高程序的性能和吞吐量。
下面我們用一個(gè)簡(jiǎn)單的例子,來(lái)解讀一下如何應(yīng)用讀寫(xiě)鎖。
public class TestReadWriteLock {
//可以同時(shí)執(zhí)行3個(gè)線(xiàn)程任務(wù)的線(xiàn)程池
ExecutorService executor = Executors.newFixedThreadPool(3);
//讀寫(xiě)目標(biāo),寫(xiě)線(xiàn)程放入數(shù)據(jù)到map,讀線(xiàn)程從map讀取數(shù)據(jù)
Map<String, String> map = new HashMap<>();
//讀寫(xiě)鎖操作對(duì)象
ReadWriteLock lock = new ReentrantReadWriteLock();
//寫(xiě)操作函數(shù)
public void write(){
executor.submit(() -> { //線(xiàn)程池提交寫(xiě)操作任務(wù)
lock.writeLock().lock(); //加寫(xiě)鎖
try {
map.put("key", "val"); //寫(xiě)數(shù)據(jù)操作
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.writeLock().unlock(); //釋放寫(xiě)鎖
}
});
}
//讀操作函數(shù)
public void read(){
lock.readLock().lock(); //加讀鎖
System.out.println(Thread.currentThread().getName() + "加讀鎖");
try {
System.out.println(map.get("key")); //讀數(shù)據(jù)操作
} finally {
lock.readLock().unlock(); //釋放讀鎖
System.out.println(Thread.currentThread().getName() + "釋放讀鎖");
}
}
}
三、讀鎖之間不互斥
我們寫(xiě)一個(gè)測(cè)試方法,通過(guò)打印輸出來(lái)理解讀寫(xiě)鎖控制代碼的執(zhí)行順序。
//測(cè)試
public static void main(String[] args) {
TestReadWriteLock test = new TestReadWriteLock();
test.write(); //提交一次寫(xiě)操作任務(wù),寫(xiě)一條數(shù)據(jù)
Runnable readTask = test::read; //線(xiàn)程方法read,實(shí)現(xiàn)線(xiàn)程Runnable接口的簡(jiǎn)便寫(xiě)法
test.executor.submit(readTask); //讀1次(新讀線(xiàn)程)
test.executor.submit(readTask); //讀2次 (新讀線(xiàn)程)
test.executor.shutdown();
}
執(zhí)行上面的代碼,可能會(huì)出現(xiàn)下面的輸出
pool-1-thread-2加讀鎖
pool-1-thread-3加讀鎖
val
val
pool-1-thread-3釋放讀鎖
pool-1-thread-2釋放讀鎖
在pool-1-thread-2沒(méi)有釋放讀鎖情況下,pool-1-thread-3可以再次加讀鎖,并且都正確的讀取到數(shù)據(jù)val。說(shuō)明讀鎖之間是不互斥的。但是,在進(jìn)行讀操作(讀鎖生效)的時(shí)候,寫(xiě)操作是無(wú)法進(jìn)行的(無(wú)法獲取寫(xiě)鎖),所以ReentrantReadWriteLock不支持同時(shí)加讀鎖和寫(xiě)鎖。 這個(gè)結(jié)論我可以負(fù)責(zé)任告訴大家,這里我就不做驗(yàn)證了!
到此這篇關(guān)于java并發(fā)編程中ReentrantLock可重入讀寫(xiě)鎖的文章就介紹到這了,更多相關(guān)java ReentrantLock可重入讀寫(xiě)鎖內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- java并發(fā)編程StampedLock高性能讀寫(xiě)鎖詳解
- Java并發(fā)之搞懂讀寫(xiě)鎖
- Java并發(fā)編程之ReadWriteLock讀寫(xiě)鎖的操作方法
- Java并發(fā)編程之重入鎖與讀寫(xiě)鎖
- Java并發(fā)編程之顯示鎖ReentrantLock和ReadWriteLock讀寫(xiě)鎖
- Java多線(xiàn)程之ReentrantReadWriteLock源碼解析
- Java多線(xiàn)程 ReentrantReadWriteLock原理及實(shí)例詳解
- 一文了解Java讀寫(xiě)鎖ReentrantReadWriteLock的使用
- 詳解Java?ReentrantReadWriteLock讀寫(xiě)鎖的原理與實(shí)現(xiàn)
- Java并發(fā)讀寫(xiě)鎖ReentrantReadWriteLock 使用場(chǎng)景
相關(guān)文章
java通過(guò)Arrays.sort(int[] a)實(shí)現(xiàn)由大到小排序的方法實(shí)現(xiàn)
Java中的Arrays.sort()方法是一種內(nèi)置的排序方法,用于對(duì)數(shù)組進(jìn)行排序,本文就來(lái)介紹一下java中的Arrays.sort()排序方法的用法,具有一定的參考價(jià)值,感興趣的可以了解一下2023-12-12
詳解java封裝實(shí)現(xiàn)Excel建表讀寫(xiě)操作
這篇文章給大家分享了java封裝實(shí)現(xiàn)Excel建表讀寫(xiě)操作的相關(guān)知識(shí)點(diǎn)內(nèi)容,有需要的朋友們可以學(xué)習(xí)下。2018-08-08
學(xué)習(xí)Java之二叉樹(shù)的編碼實(shí)現(xiàn)過(guò)程詳解
本文將通過(guò)代碼來(lái)進(jìn)行二叉樹(shù)的編碼實(shí)現(xiàn),文中的代碼示例介紹的非常詳細(xì),對(duì)我們學(xué)習(xí)Java二叉樹(shù)有一定的幫助,感興趣的同學(xué)跟著小編一起來(lái)看看吧2023-08-08
java教程之java程序編譯運(yùn)行圖解(java程序運(yùn)行)
最近重新復(fù)習(xí)了一下java基礎(chǔ),在使用javap的過(guò)程中遇到了一些問(wèn)題,這里便講講對(duì)于一個(gè)類(lèi)文件如何編譯、運(yùn)行、反編譯的。也讓自己加深一下印象2014-03-03
MybatisPlus實(shí)現(xiàn)簡(jiǎn)單增刪改查功能
這篇文章主要介紹了MybatisPlus實(shí)現(xiàn)簡(jiǎn)單增刪改查的相關(guān)資料,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04
JVM運(yùn)行時(shí)數(shù)據(jù)區(qū)原理解析
這篇文章主要介紹了JVM運(yùn)行時(shí)數(shù)據(jù)區(qū)原理解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08
SpringBoot程序的打包與運(yùn)行的實(shí)現(xiàn)
本文主要介紹了SpringBoot程序的打包與運(yùn)行的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06
springboot與vue詳解實(shí)現(xiàn)短信發(fā)送流程
隨著人工智能的不斷發(fā)展,機(jī)器學(xué)習(xí)這門(mén)技術(shù)也越來(lái)越重要,很多人都開(kāi)啟了學(xué)習(xí)機(jī)器學(xué)習(xí),本文就介紹了機(jī)器學(xué)習(xí)的基礎(chǔ)內(nèi)容2022-06-06

