亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java中ReentrantReadWriteLock讀寫(xiě)鎖的實(shí)現(xiàn)

 更新時(shí)間:2025年05月23日 08:33:49   作者:Volunteer Technology  
本文主要介紹了ReentrantReadWriteLock讀寫(xiě)鎖,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

一、鎖的分類

這里不會(huì)對(duì)Java中大部分的分類都聊清楚,主要把 **互斥,共享** 這種分類聊清楚。

Java中的互斥鎖,synchronized,ReentrantLock這種都是互斥鎖。一個(gè)線程持有鎖操作時(shí),其他線程都需要等待前面的線程釋放鎖資源,才能重新嘗試競(jìng)爭(zhēng)這把鎖。

Java中的讀寫(xiě)鎖(支撐互斥&共享),Java中最常見(jiàn)的就是 **ReentrantReadWriteLock** ,StampedLock。

其中StampedLock是JDK1.8中推出的一款讀寫(xiě)鎖的實(shí)現(xiàn),針對(duì)ReentrantReadWriteLock一個(gè)優(yōu)化。但是,今兒不細(xì)聊。主要玩ReentrantReadWriteLock。

ReentrantReadWriteLock主要就是解決咱們剛才聊的,讀寫(xiě)操作都有,讀操作居多,寫(xiě)操作頻次相對(duì)比較低的情況,可以使用讀寫(xiě)鎖來(lái)提升系統(tǒng)性能。

讀寫(xiě)鎖中:

* 寫(xiě)寫(xiě)互斥
* 讀寫(xiě)互斥
* 寫(xiě)讀互斥
* 讀讀共享
* 有鎖降級(jí)的情況,后面聊?。?/p>

二、ReentrantReadWriteLock的基本操作

ReentrantReadWriteLock中實(shí)現(xiàn)了ReadWriteLock的接口,在這個(gè)接口里面提供了兩個(gè)抽象方法。

正常的操作,是new ReentrantReadWriteLock的對(duì)象,但是你具體的業(yè)務(wù)操作是需要讀鎖,還是寫(xiě)鎖,你需要單獨(dú)的獲取到,然后針對(duì)性的加鎖。

public interface ReadWriteLock {
    /**
     * Returns the lock used for reading.
     *
     * @return the lock used for reading
     */
    Lock readLock();

    /**
     * Returns the lock used for writing.
     *
     * @return the lock used for writing
     */
    Lock writeLock();
}

具體使用方式

public static void main(String[] args){
    // 1、構(gòu)建讀寫(xiě)鎖對(duì)象
    ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();

    // 2、單獨(dú)獲取讀、寫(xiě)鎖對(duì)象
    ReentrantReadWriteLock.ReadLock readLock = readWriteLock.readLock();
    ReentrantReadWriteLock.WriteLock writeLock = readWriteLock.writeLock();

    // 3、根據(jù)業(yè)務(wù)使用具體的鎖對(duì)象加鎖
    writeLock.lock();
    // try-finally的目的,是為了避免沒(méi)有及時(shí)釋放鎖資源導(dǎo)致死鎖的問(wèn)題。
    try{
        // 4、業(yè)務(wù)操作…………
        System.out.println("寫(xiě)操作");
    }finally {
        // 5、釋放鎖
        writeLock.unlock();
    }
}

三、ReentrantReadWriteLock的底層實(shí)現(xiàn)

ReentrantReadWriteLock是基于AQS實(shí)現(xiàn)的。

AQS是JUC包下的一個(gè)抽象類AbstractQueuedSynchronizer

暫時(shí)只關(guān)注兩點(diǎn),分別是AQS提供的state屬性,還有AQS提供的一個(gè)同步隊(duì)列。

state屬性,用來(lái)標(biāo)識(shí)當(dāng)前 讀寫(xiě)鎖 的資源是否被占用的核心標(biāo)識(shí)。
private volatile int state;

一個(gè)int類型的state,是4字節(jié),每個(gè)字節(jié)占用8個(gè)bit位,一個(gè)state占用32個(gè)bit位。

* 高16位,作為讀鎖的標(biāo)記。
* 低16位,作為寫(xiě)鎖的標(biāo)記。

static final int SHARED_SHIFT   = 16;
static final int EXCLUSIVE_MASK = (1 << SHARED_SHIFT) - 1;
00000000 00000000 11111111 11111111
/** 查看讀鎖的占用情況。 */
static int sharedCount(int state)    { return state >>> SHARED_SHIFT; }
/** Returns the number of exclusive holds represented in count  */
static int exclusiveCount(int state) { return state & EXCLUSIVE_MASK; }

00000000 00000000 00000000 00000000    int類型的數(shù)值的32個(gè)bit位。 

讀鎖占用情況:
00000000 00000011 00000000 00000000    state
>>> 16 
00000000 00000000 00000000 00000011    讀鎖被獲取了三次。

寫(xiě)鎖占用情況。(這里之所以&這個(gè)么東西,是對(duì)后期的鎖降級(jí)有影響~)
00000000 00000000 00000000 00000001    state
&
00000000 00000000 11111111 11111111  
=
00000000 00000000 00000000 00000001    寫(xiě)鎖被獲取了一次。

一個(gè)同步隊(duì)列,當(dāng)線程獲取鎖資源失敗時(shí),需要到這個(gè)同步隊(duì)列中排隊(duì)。到了合適的時(shí)機(jī),就會(huì)繼續(xù)嘗試獲取對(duì)應(yīng)的鎖資源。

四、ReentrantReadWriteLock的鎖重入

同一個(gè)線程,多次獲取同一把鎖時(shí),就會(huì)出現(xiàn)鎖重入的情況。

而咱們大多數(shù)的鎖,都會(huì)提供鎖重入的功能。

鎖重入場(chǎng)景:

public class Demo {
    // 1、構(gòu)建讀寫(xiě)鎖對(duì)象
    static ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();

    // 2、單獨(dú)獲取讀、寫(xiě)鎖對(duì)象
    static ReentrantReadWriteLock.ReadLock readLock;
    static ReentrantReadWriteLock.WriteLock writeLock;

    static{
        // 2、單獨(dú)獲取讀、寫(xiě)鎖對(duì)象
        readLock = readWriteLock.readLock();
        writeLock = readWriteLock.writeLock();
    }

    public static void main(String[] args){
        // 3、根據(jù)業(yè)務(wù)使用具體的鎖對(duì)象加鎖
        writeLock.lock();
        // try-finally的目的,是為了避免沒(méi)有及時(shí)釋放鎖資源導(dǎo)致死鎖的問(wèn)題。
        try{
            // 4、業(yè)務(wù)操作…………調(diào)用其他方法
            xxx();
        }finally {
            // 5、釋放鎖
            writeLock.unlock();
        }
    }

    private static void xxx(){
        writeLock.lock();
        try{
            // 其他按業(yè)務(wù)
        }finally {
            writeLock.unlock();
        }
    }
}

咱們底層的鎖重入邏輯很簡(jiǎn)單

**寫(xiě)鎖:** 寫(xiě)鎖的實(shí)現(xiàn)就是每一次獲取寫(xiě)鎖時(shí),會(huì)對(duì)state的低16位+1,再次獲取,再次+1。同理,每次釋放鎖資源時(shí),也需要對(duì)state進(jìn)行-1。 而當(dāng)對(duì)state的低16位減到0時(shí),鎖資源就釋放干凈了。

讀鎖: 首先,讀鎖是共享的,他用state的高16位來(lái)維護(hù)信息。如果高16位的state的值,經(jīng)過(guò)運(yùn)算,知道了是4,也就是讀鎖被獲取了4次??赡蹵線程獲取了2次讀鎖資源。 B線程獲取了2次讀鎖資源。高位的state自然就是4。但是因?yàn)槌绦騿T寫(xiě)代碼除了問(wèn)題,使用A線程,釋放了4次讀鎖資源,那此時(shí)B線程是不是就可能出現(xiàn)數(shù)據(jù)安全問(wèn)題了。

所以,為了解決上述的問(wèn)題,每個(gè)線程需要獨(dú)立的記錄自己獲取了幾次讀鎖資源??梢允褂肨hreadLocal來(lái)保存線程局部的信息,每次加鎖時(shí),ThreadLocal中需要存儲(chǔ)一個(gè)標(biāo)記,每次+1。每次釋放鎖時(shí),也需要將ThreadLocal中的標(biāo)記進(jìn)行-1。讀線程最后是基于自己的ThreadLocal中的數(shù)值,來(lái)確認(rèn)讀鎖是否釋放干凈。

五、ReentrantReadWriteLock的寫(xiě)鎖饑餓

寫(xiě)鎖饑餓的問(wèn)題。

如果寫(xiě)線程在AQS中排隊(duì),并且排在head.next的位置。 那么其他想獲取讀鎖的讀線程需要排隊(duì)。避免大量的讀請(qǐng)求獲取讀鎖,讓寫(xiě)線程一直AQS隊(duì)列中排隊(duì),無(wú)法執(zhí)行寫(xiě)操作的問(wèn)題。

通過(guò)源碼可以看到,讀寫(xiě)鎖中,僅僅針對(duì)head.next這個(gè)節(jié)點(diǎn)的情況,來(lái)確認(rèn)讀線程獲取讀鎖時(shí)是否需要排隊(duì)

// 這個(gè)方法,總結(jié)一句話。  
// AQS中有排隊(duì)的Node,并且head的next節(jié)點(diǎn)是一個(gè)有線程并且在等待寫(xiě)鎖的Node
final boolean apparentlyFirstQueuedIsExclusive() {
    Node h, s;
    return (h = head) != null &&
        (s = h.next)  != null &&
        !s.isShared()         &&
        s.thread != null;
}

ReentrantReadWriteLock讀寫(xiě)鎖中有鎖降級(jí),但是這個(gè)和synchronized的鎖升級(jí)沒(méi)任何關(guān)系?。?!

六、ReentrantReadWriteLock的鎖降級(jí)

ReentrantReadWriteLock的鎖降級(jí)是指當(dāng)前線程如果持有了寫(xiě)鎖,可以降級(jí)直接獲取到讀鎖。

在讀寫(xiě)鎖中,持有寫(xiě)鎖的同時(shí),再去獲取讀鎖,這種行為一般被稱為 **鎖降級(jí)** 。

在讀寫(xiě)鎖中,持有讀鎖的同時(shí),去獲取寫(xiě)鎖,這種行為被稱為 **鎖升級(jí)** ,這個(gè)行為是不允許的。

這里是獲取讀鎖的的邏輯,看一下鎖降級(jí)的支持方式

// 競(jìng)爭(zhēng)讀鎖。
if (exclusiveCount(c) != 0 &&   // 這行代表某個(gè)線程持有寫(xiě)鎖
    getExclusiveOwnerThread() != current)    // 這行代表持有寫(xiě)鎖的不是當(dāng)前線程
    // 退出競(jìng)爭(zhēng),無(wú)法獲取讀鎖
    return -1;  

前面邏輯沒(méi)有走return - 1之后,在后續(xù)就會(huì)正常的對(duì)state的高位+1,并且完成讀鎖的計(jì)數(shù)操作。

七、ReentrantReadWriteLock的優(yōu)化 

ReentrantReadWriteLock的優(yōu)化主要是在讀鎖計(jì)數(shù)層面上做的優(yōu)化。

這個(gè)對(duì)性能的優(yōu)化微乎其微,但是確確實(shí)實(shí)是一個(gè)優(yōu)化。

在獲取讀鎖時(shí),因?yàn)槭枪蚕淼?,這種優(yōu)化只針對(duì)第一個(gè)獲取讀鎖的線程和最后一個(gè)獲取讀鎖的線程。

針對(duì)第一個(gè)獲取讀鎖的線程,他采用一個(gè)全局變量記錄重入次數(shù)。這個(gè)操作可以節(jié)省掉使用ThreadLocal的時(shí)間成本和內(nèi)存成本。

其中firstReader記錄第一個(gè)獲取讀鎖的線程。

firstReaderHoldCount,記錄第一個(gè)獲取讀鎖的線程的重入次數(shù)。

這里是最后一個(gè)獲取讀鎖的線程需要走的邏輯

cachedHoldCounter這個(gè)屬性是記錄最后一個(gè)獲取讀鎖的線程的重入次數(shù)。

這里可以讓最后一個(gè)獲取讀鎖的線程在重入時(shí),省略掉去ThreadLocal中g(shù)et計(jì)數(shù)器的操作,但是之前的set存儲(chǔ)操作,不能省略

// 獲取上次最后獲取讀鎖的線程
HoldCounter rh = cachedHoldCounter;
// 查看當(dāng)前線程是否是之前的cachedHoldCounter
if (rh == null || rh.tid != getThreadId(current))
    // 說(shuō)明不是,將當(dāng)前獲取讀鎖的線程設(shè)置為cachedHoldCounter
    cachedHoldCounter = rh = readHolds.get();
// 這個(gè)判斷代表第一次獲取讀鎖才會(huì)進(jìn)去
else if (rh.count == 0)
    // 如果是第一次獲取讀鎖,不是重入,還是需要扔到ThreadLocal里紀(jì)錄好,。
    readHolds.set(rh);
// 直接對(duì)獲取到的rh做++操作,代表獲取了一次讀鎖。
rh.count++;

到此這篇關(guān)于Java中ReentrantReadWriteLock讀寫(xiě)鎖的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)ReentrantReadWriteLock讀寫(xiě)鎖內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 基于SpringBoot制作一個(gè)PDF切圖小工具

    基于SpringBoot制作一個(gè)PDF切圖小工具

    這篇文章主要為大家詳細(xì)介紹了如何基于SpringBoot制作一個(gè)PDF切圖小工具,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2024-01-01
  • java list常用方法總結(jié)

    java list常用方法總結(jié)

    這篇文章主要介紹了java list常用方法總結(jié),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • 將Java(SpringBoot)項(xiàng)目打包為Docker鏡像的三種方法

    將Java(SpringBoot)項(xiàng)目打包為Docker鏡像的三種方法

    這篇文章主要介紹了將Java(SpringBoot)項(xiàng)目打包為Docker鏡像的三種方法,分別是手動(dòng)構(gòu)建、使用Dockerfile和使用SpringBootMaven插件,每種方法都有其特點(diǎn)和適用場(chǎng)景,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2025-03-03
  • Java連接mysql數(shù)據(jù)庫(kù)以及mysql驅(qū)動(dòng)jar包下載和使用方法

    Java連接mysql數(shù)據(jù)庫(kù)以及mysql驅(qū)動(dòng)jar包下載和使用方法

    這篇文章主要給大家介紹了關(guān)于Java連接mysql數(shù)據(jù)庫(kù)以及mysql驅(qū)動(dòng)jar包下載和使用方法,MySQL是一款常用的關(guān)系型數(shù)據(jù)庫(kù),它的JDBC驅(qū)動(dòng)程序使得我們可以通過(guò)Java程序連接MySQL數(shù)據(jù)庫(kù)進(jìn)行數(shù)據(jù)操作,需要的朋友可以參考下
    2023-11-11
  • Java比較兩個(gè)對(duì)象大小的三種方法詳解

    Java比較兩個(gè)對(duì)象大小的三種方法詳解

    在優(yōu)先級(jí)隊(duì)列中插入的元素必須能比較大小,如果不能比較大小,如插入兩個(gè)學(xué)生類型的元素,會(huì)報(bào)ClassCastException異常。本文就為大家總結(jié)了Java比較兩個(gè)對(duì)象大小的三種方法,需要的可以參考一下
    2022-07-07
  • Java8 實(shí)現(xiàn)stream將對(duì)象集合list中抽取屬性集合轉(zhuǎn)化為map或list

    Java8 實(shí)現(xiàn)stream將對(duì)象集合list中抽取屬性集合轉(zhuǎn)化為map或list

    這篇文章主要介紹了Java8 實(shí)現(xiàn)stream將對(duì)象集合list中抽取屬性集合轉(zhuǎn)化為map或list的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-02-02
  • Maven包沖突導(dǎo)致NoSuchMethodError錯(cuò)誤的解決辦法

    Maven包沖突導(dǎo)致NoSuchMethodError錯(cuò)誤的解決辦法

    web 項(xiàng)目 能正常編譯,運(yùn)行時(shí)也正常啟動(dòng),但執(zhí)行到需要調(diào)用 org.codehaus.jackson 包中的某個(gè)方法時(shí),產(chǎn)生運(yùn)行異常,這篇文章主要介紹了Maven包沖突導(dǎo)致NoSuchMethodError錯(cuò)誤的解決辦法,需要的朋友可以參考下
    2024-05-05
  • SpringBoot實(shí)現(xiàn)阿里云快遞物流查詢的示例代碼

    SpringBoot實(shí)現(xiàn)阿里云快遞物流查詢的示例代碼

    本文將基于springboot實(shí)現(xiàn)快遞物流查詢,物流信息的獲取通過(guò)阿里云第三方實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2021-10-10
  • java  文件鎖的簡(jiǎn)單實(shí)現(xiàn)

    java 文件鎖的簡(jiǎn)單實(shí)現(xiàn)

    這篇文章主要介紹了java 文件鎖的簡(jiǎn)單實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下
    2017-07-07
  • nodejs與JAVA應(yīng)對(duì)高并發(fā)的對(duì)比方式

    nodejs與JAVA應(yīng)對(duì)高并發(fā)的對(duì)比方式

    這篇文章主要介紹了nodejs與JAVA應(yīng)對(duì)高并發(fā)的對(duì)比方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-08-08

最新評(píng)論