成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

[JDK源碼]J.U.C-AQS-ReentrantLock

不知名網(wǎng)友 / 2252人閱讀

摘要:公平鎖阻塞隊(duì)列前邊有線程,要去后邊排隊(duì),簡單來說滾后邊等著去。非公平鎖不管是否有線程排隊(duì),先槍鎖基于實(shí)現(xiàn)的可重入鎖實(shí)現(xiàn)類。

AQS原理介紹:

AQS (AbstractQueuedSynchronizer)底層一個(gè)隊(duì)列 阻塞隊(duì)列 ->
? Abstract:因?yàn)樗⒉恢涝趺瓷湘i。模板方法設(shè)計(jì)模式即可,暴露出鎖邏輯。
? Queue :線程阻塞隊(duì)列 Synchronizer:同步
? CAS + state 完成多線程槍鎖邏輯 Queue 完成搶不到鎖的線程排隊(duì)

AQS核心代碼

//獲取鎖public final void acquire(int arg) {    if (!tryAcquire(arg) && // 子類判定獲取鎖是否失敗        acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) // 獲取失敗后添加到阻塞隊(duì)列        selfInterrupt();}// 子類實(shí)現(xiàn)獲取鎖的邏輯,AQS并不知道你怎么用這個(gè)state來上鎖protected boolean tryAcquire(int arg) {    throw new UnsupportedOperationException();}//釋放鎖的public final boolean release(int arg) {    // 子類判定釋放鎖成功    if (tryRelease(arg)) {        // 檢查阻塞隊(duì)列喚醒即可        Node h = head;        if (h != null && h.waitStatus != 0)            unparkSuccessor(h);        return true;    }    return false;}// 子類實(shí)現(xiàn)獲取鎖的邏輯protected boolean tryRelease(int arg) {    throw new UnsupportedOperationException();}

子類只需要實(shí)現(xiàn)自己的獲取鎖邏輯和釋放鎖邏輯即可,至于排隊(duì)阻塞等待、喚醒機(jī)制均由AQS來完成。

公平鎖:阻塞隊(duì)列前邊有線程,要去后邊排隊(duì),簡單來說 滾后邊等著去。(FIFO) 非公平鎖:不管是否有線程排隊(duì),先槍鎖

ReentrantLock

基于AQS實(shí)現(xiàn)的可重入鎖實(shí)現(xiàn)類。
ReentrantLock 是一個(gè)可重入的互斥(/獨(dú)占)鎖,又稱為“獨(dú)占鎖”
重入鎖: 自己可以再次獲取自己的內(nèi)部的鎖。
state 來表示鎖狀態(tài)和重入次數(shù),0無鎖,大于0 重入次數(shù),1為重入1次,也即只加鎖一次

核心屬性

private final Sync sync;//sync,構(gòu)造方法中初始化,決定使用公平鎖還是非公平鎖的方式獲取鎖。

構(gòu)造方法

	public ReentrantLock() {        // 默認(rèn)為非公平鎖。        sync = new NonfairSync();    }	public ReentrantLock(boolean fair) {        // 由fair變量來表明選擇鎖類型        sync = fair ? new FairSync() : new NonfairSync();    }

核心內(nèi)部類

abstract static class Sync extends AbstractQueuedSynchronizer {}
static final class NonfairSync extends Sync {} 非公平鎖
static final class FairSync extends Sync {} 公平鎖

            //非公平鎖        static final class NonfairSync extends Sync {            // 由ReentrantLock調(diào)用獲取鎖            final void lock() {                // 非公平鎖,直接搶鎖,不管有沒有線程排隊(duì)                if (compareAndSetState(0, 1))                    // 上鎖成功,標(biāo)識(shí)當(dāng)前線程為獲取鎖的線程                    setExclusiveOwnerThread(Thread.currentThread());                else                    // 搶鎖失敗,進(jìn)入AQS的標(biāo)準(zhǔn)獲取鎖流程                    acquire(1);            }            protected final boolean tryAcquire(int acquires) {                // 使用父類提供的獲取非公平鎖的方法來獲取鎖                return nonfairTryAcquire(acquires);            }        }        //公平鎖        static final class FairSync extends Sync {            // 由ReentrantLock調(diào)用            final void lock() {                // 沒有嘗試搶鎖,直接進(jìn)入AQS標(biāo)準(zhǔn)獲取鎖流程                acquire(1);            }            // AQS調(diào)用,子類自己實(shí)現(xiàn)獲取鎖的流程            protected final boolean tryAcquire(int acquires) {                final Thread current = Thread.currentThread();                int c = getState();                // 此時(shí)有可能正好獲取鎖的線程釋放了鎖,也有可能本身就沒有線程獲取鎖                if (c == 0) {                   //這里和非公平鎖的區(qū)別在于:hasQueuedPredecessors看看隊(duì)列中是否有線程正在排隊(duì),沒有的話再通過CAS搶鎖                    if (!hasQueuedPredecessors() &&                            compareAndSetState(0, acquires)) {                        setExclusiveOwnerThread(current);                        return true;                    }                }                // 當(dāng)前線程就是獲取鎖的線程,那么這里是鎖重入,和非公平鎖操作一模一樣                else if (current == getExclusiveOwnerThread()) {                    int nextc = c + acquires;                    if (nextc < 0)                        throw new Error("Maximum lock count exceeded");                    setState(nextc);                    return true;                }                //需要AQS來將當(dāng)前線程放入阻塞隊(duì)列,然后進(jìn)行阻塞操作等待喚醒獲取鎖                return false;            }        }        abstract static class Sync extends AbstractQueuedSynchronizer {            abstract void lock();            // 非公平鎖標(biāo)準(zhǔn)獲取鎖方法            final boolean nonfairTryAcquire(int acquires) {                final Thread current = Thread.currentThread();                int c = getState();                //獲取所得線程釋放了鎖,那么可以嘗試搶鎖                if (c == 0) {                    // 繼續(xù)搶鎖,不看有沒有線程排隊(duì)                    if (compareAndSetState(0, acquires)) {                        setExclusiveOwnerThread(current);                        return true;                    }                }                // 當(dāng)前線程就是持有鎖的線程,表明鎖重入                else if (current == getExclusiveOwnerThread()) {                    // 利用state 進(jìn)行次數(shù)記錄                    int nextc = c + acquires;                    // 如果超過了int表示范圍,表明符號(hào)溢出,所以拋出異常                    if (nextc < 0)                        throw new Error("Maximum lock count exceeded");                    setState(nextc);//state 變量賦值                    return true;                }                //表明需要AQS來將當(dāng)前線程放入阻塞隊(duì)列,然后進(jìn)行阻塞操作等待喚醒獲取鎖                return false;            }            // 公平鎖和非公平鎖公用方法,在釋放鎖的時(shí)候,并不區(qū)分是否公平            protected final boolean tryRelease(int releases) {                int c = getState() - releases;                // 如果當(dāng)前線程不是上鎖的那個(gè)線程                if (Thread.currentThread() != getExclusiveOwnerThread())                    throw new IllegalMonitorStateException();                boolean free = false;                // 不是重入鎖,那么當(dāng)前線程一定是釋放鎖了,把當(dāng)前AQS用于保存當(dāng)前鎖對(duì)象的變量ExclusiveOwnerThread設(shè)置為null,表明釋放鎖成功                if (c == 0) {                    free = true;                    setExclusiveOwnerThread(null);                }                //此時(shí)state全局變量沒有改變,也就意味著在setState之前                //沒有別的線程能夠獲取鎖,保證了以上的操作原子性                setState(c);                //釋放鎖成功了,可以去喚醒正在等待鎖的線程                return free;            }            protected final boolean isHeldExclusively() {                return getExclusiveOwnerThread() == Thread.currentThread();            }            final ConditionObject newCondition() {                return new ConditionObject();            }        }

核心方法

public void lock() {//獲取鎖的操作    // 直接通過sync同步器上鎖    sync.lock();}public void unlock() {//釋放鎖的操作    sync.release(1);}

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/125657.html

相關(guān)文章

  • java源碼

    摘要:集合源碼解析回歸基礎(chǔ),集合源碼解析系列,持續(xù)更新和源碼分析與是兩個(gè)常用的操作字符串的類。這里我們從源碼看下不同狀態(tài)都是怎么處理的。 Java 集合深入理解:ArrayList 回歸基礎(chǔ),Java 集合深入理解系列,持續(xù)更新~ JVM 源碼分析之 System.currentTimeMillis 及 nanoTime 原理詳解 JVM 源碼分析之 System.currentTimeMi...

    Freeman 評(píng)論0 收藏0
  • HashSet源碼分析:JDK源碼系列

    摘要:簡介繼續(xù)分析源碼,上一篇文章把的分析完畢。本文開始分析簡單的介紹一下。存儲(chǔ)的元素是無序的并且允許使用空的元素。 1.簡介 繼續(xù)分析源碼,上一篇文章把HashMap的分析完畢。本文開始分析HashSet簡單的介紹一下。 HashSet是一個(gè)無重復(fù)元素集合,內(nèi)部使用HashMap實(shí)現(xiàn),所以HashMap的特征耶繼承了下來。存儲(chǔ)的元素是無序的并且HashSet允許使用空的元素。 HashSe...

    用戶83 評(píng)論0 收藏0
  • HashMap 源碼詳細(xì)分析(JDK1.8)

    摘要:則使用了拉鏈?zhǔn)降纳⒘兴惴ǎ⒃谥幸肓思t黑樹優(yōu)化過長的鏈表。如果大家對(duì)紅黑樹感興趣,可以閱讀我的另一篇文章紅黑樹詳細(xì)分析。構(gòu)造方法構(gòu)造方法分析的構(gòu)造方法不多,只有四個(gè)。 1.概述 本篇文章我們來聊聊大家日常開發(fā)中常用的一個(gè)集合類 - HashMap。HashMap 最早出現(xiàn)在 JDK 1.2中,底層基于散列算法實(shí)現(xiàn)。HashMap 允許 null 鍵和 null 值,在計(jì)算哈鍵的哈希值...

    monw3c 評(píng)論0 收藏0
  • 集合框架源碼學(xué)習(xí)之HashMap(JDK1.8)

    摘要:所謂拉鏈法就是將鏈表和數(shù)組相結(jié)合。若遇到哈希沖突,則將沖突的值加到鏈表中即可。在編寫程序中,要盡量避免。 目錄: 0-1. 簡介 0-2. 內(nèi)部結(jié)構(gòu)分析   0-2-1. JDK18之前   0-2-2. JDK18之后 0-3. LinkedList源碼分析   0-3-1. 構(gòu)造方法   0-3-2. put方法   0-3-3. get方法   0-3-4. resize方法 ...

    yangrd 評(píng)論0 收藏0
  • JDK1.8源碼分析01之學(xué)習(xí)建議(可以延伸其他源碼學(xué)習(xí))

    摘要:唐老師,回答道讀源碼是要建立在你的基礎(chǔ)經(jīng)驗(yàn)足夠的情況下。除了自己去閱讀源碼之外,比如學(xué)習(xí)某個(gè)類的時(shí)候,可以專門結(jié)合一些優(yōu)質(zhì)的博客針對(duì)性的對(duì)比學(xué)習(xí),并查漏補(bǔ)缺。制定源碼學(xué)習(xí)計(jì)劃。多調(diào)試,跟蹤源碼。如若有好的學(xué)習(xí)方法,可以留言一起交流學(xué)習(xí)。 序言:目前看一看源碼,來提升自己的技術(shù)實(shí)力。同時(shí)現(xiàn)在好多面試官都喜歡問源碼,問你是否讀過JDK源碼等等? 針對(duì)如何閱讀源碼,也請(qǐng)教了我的老師。下面就先...

    ky0ncheng 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<