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

Java中的DelayQueue源碼解析

 更新時(shí)間:2023年12月13日 08:32:59   作者:demon7552003  
這篇文章主要介紹了Java中的DelayQueue源碼解析,一個(gè)實(shí)現(xiàn)PriorityBlockingQueue實(shí)現(xiàn)延遲獲取的無界隊(duì)列,在創(chuàng)建元素時(shí),可以指定多久才能從隊(duì)列中獲取當(dāng)前元素,只有延時(shí)期滿后才能從隊(duì)列中獲取元素,需要的朋友可以參考下

介紹

一個(gè)實(shí)現(xiàn)PriorityBlockingQueue實(shí)現(xiàn)延遲獲取的無界隊(duì)列,在創(chuàng)建元素時(shí),可以指定多久才能從隊(duì)列中獲取當(dāng)前元素。只有延時(shí)期滿后才能從隊(duì)列中獲取元素。

DelayQueue可以運(yùn)用在以下應(yīng)用場景:

1.緩存系統(tǒng)的設(shè)計(jì):可以用DelayQueue保存緩存元素的有效期,使用一個(gè)線程循環(huán)查詢DelayQueue,一旦能從DelayQueue中獲取元素時(shí),表示緩存有效期到了。

2.定時(shí)任務(wù)調(diào)度。使用DelayQueue保存當(dāng)天將會(huì)執(zhí)行的任務(wù)和執(zhí)行時(shí)間,一旦從DelayQueue中獲取到任務(wù)就開始執(zhí)行,從比如TimerQueue就是使用DelayQueue實(shí)現(xiàn)的。

數(shù)據(jù)結(jié)構(gòu)

public interface Delayed extends Comparable<Delayed> {
 
    /**
     * 返回與此對象相關(guān)的剩余延遲時(shí)間,以給定的時(shí)間單位表示
     */
    long getDelay(TimeUnit unit);
}

 getDelay方法一般用內(nèi)部存儲的事件,減去當(dāng)前事件,即為剩余延遲事件

屬性

  private final transient ReentrantLock lock = new ReentrantLock();
    private final PriorityQueue<E> q = new PriorityQueue<E>();
 
    /**
    *用于優(yōu)化內(nèi)部阻塞通知的線程
     */
    private Thread leader = null;
    private final Condition available = lock.newCondition();

以支持優(yōu)先級的PriorityQueue無界隊(duì)列作為一個(gè)容器,因?yàn)樵囟急仨殞?shí)現(xiàn)Delayed接口,可以根據(jù)元素的過期時(shí)間來對元素進(jìn)行排列,因此,先過期的元素會(huì)在隊(duì)首,每次從隊(duì)列里取出來都是最先要過期的元素。

leader是一個(gè)Thread元素,它在offer和take中都有使用,它代表當(dāng)前獲取到鎖的消費(fèi)者線程,

DelayQueue實(shí)現(xiàn)Leader-Folloer pattern

1、當(dāng)存在多個(gè)take線程時(shí),同時(shí)只生效一個(gè),即,leader線程

2、當(dāng)leader存在時(shí),其它的take線程均為follower,其等待是通過condition實(shí)現(xiàn)的

3、當(dāng)leader不存在時(shí),當(dāng)前線程即成為leader,在delay之后,將leader角色釋放還原

4、最后如果隊(duì)列還有內(nèi)容,且leader空缺,則調(diào)用一次condition的signal,喚醒掛起的take線程,其中之一將成為新的leader

5、最后在finally中釋放鎖

方法實(shí)現(xiàn)

offer,poll,peek

    public boolean offer(E e) {
        final ReentrantLock lock = this.lock;
        lock.lock();
        try {
            q.offer(e);
            //如果插入元素是第一個(gè)元素
            if (q.peek() == e) {
                //leader設(shè)置為null
                leader = null;
                //喚醒
                available.signal();
            }
            return true;
        } finally {
            lock.unlock();
        }
    }
    public boolean offer(E e, long timeout, TimeUnit unit) {
        return offer(e);
    }
    public E poll() {
        final ReentrantLock lock = this.lock;
        lock.lock();
        try {
            E first = q.peek();
            //如果未到期,則返回null,否則刪除
            if (first == null || first.getDelay(NANOSECONDS) > 0)
                return null;
            else
                return q.poll();
        } finally {
            lock.unlock();
        }
    }
   public E poll(long timeout, TimeUnit unit) throws InterruptedException {
        long nanos = unit.toNanos(timeout);
        final ReentrantLock lock = this.lock;
        lock.lockInterruptibly();
        try {
            for (;;) {
                E first = q.peek();
                if (first == null) {
                    if (nanos <= 0)
                        return null;
                    else
                        nanos = available.awaitNanos(nanos);
                } else {
                    long delay = first.getDelay(NANOSECONDS);
                    //到期,則poll
                    if (delay <= 0)
                        return q.poll();
                    if (nanos <= 0)
                        return null;
                    first = null; // don't retain ref while waiting
                    if (nanos < delay || leader != null)//nanos<delay,表示超時(shí)剩余時(shí)間小于到期時(shí)間,
                        nanos = available.awaitNanos(nanos);
                    else {
                        Thread thisThread = Thread.currentThread();
                        //設(shè)置當(dāng)前線程為leader
                        leader = thisThread;
                        try {
                            //等待條件
                            long timeLeft = available.awaitNanos(delay);
                            //剩余超時(shí)時(shí)間
                            nanos -= delay - timeLeft;
                        } finally {
                            if (leader == thisThread)
                                leader = null;
                        }
                    }
                }
            }
        } finally {
            if (leader == null && q.peek() != null)
                available.signal();
            lock.unlock();
        }
    }
    public E peek() {
        final ReentrantLock lock = this.lock;
        lock.lock();
        try {
            return q.peek();
        } finally {
            lock.unlock();
        }
    }

put,take

/**
     * Retrieves and removes the head of this queue, waiting if necessary
     * until an element with an expired delay is available on this queue.
     *
     * @return the head of this queue
     * @throws InterruptedException {@inheritDoc}
     */
    public E take() throws InterruptedException {
        final ReentrantLock lock = this.lock;
        // 獲取可中斷鎖。
        lock.lockInterruptibly();
        try {
            for (;;) {
                // 從優(yōu)先級隊(duì)列中獲取隊(duì)列頭元素
                E first = q.peek();
                if (first == null)
                    // 無元素,當(dāng)前線程加入等待隊(duì)列,并阻塞
                    available.await();
                else {
                    // 通過getDelay 方法獲取延遲時(shí)間
                    long delay = first.getDelay(NANOSECONDS);
                    if (delay <= 0)
                        // 延遲時(shí)間到期,獲取并刪除頭部元素。
                        return q.poll();
                    first = null; // don't retain ref while waiting
                    if (leader != null)
                        available.await();
                    else {
                        Thread thisThread = Thread.currentThread();
                        leader = thisThread;
                        try {
                            // 線程節(jié)點(diǎn)進(jìn)入等待隊(duì)列 x 納秒。
                            available.awaitNanos(delay);
                        } finally {
                            if (leader == thisThread)
                                leader = null;
                        }
                    }
                }
            }
        } finally {
            // leader == null且還存在元素的話,喚醒一個(gè)消費(fèi)線程。
            if (leader == null && q.peek() != null)
                available.signal();
            lock.unlock();
        }
    }
   public void put(E e) {
        offer(e);
    }

take()方法邏輯:

1.獲取鎖

2.取出優(yōu)先級隊(duì)列q的首元素

3.如果元素q的隊(duì)首/隊(duì)列為空,阻塞

4.如果元素q的隊(duì)首(first)不為空,獲得這個(gè)元素的delay時(shí)間值,如果first的延遲delay時(shí)間值為0的話,說明該元素已經(jīng)到了可以使用的時(shí)間,調(diào)用poll方法彈出該元素,跳出方法

5.如果first的延遲delay時(shí)間值不為0的話,釋放元素first的引用,避免內(nèi)存泄露

6.循環(huán)以上操作,直到return

leader作用

如果leader不為null,說明已經(jīng)有消費(fèi)者線程拿到鎖,直接阻塞當(dāng)前線程,如果leader為null,把當(dāng)前線程賦值給leader,并等待剩余的到期時(shí)間,最后釋放leader,這里我們想象著我們有個(gè)多個(gè)消費(fèi)者線程用take方法去取,如果沒有l(wèi)eader!=null的判斷,這些線程都會(huì)無限循環(huán),直到返回第一個(gè)元素,很顯然很浪費(fèi)資源。所以leader的作用是設(shè)置一個(gè)標(biāo)記,來避免消費(fèi)者的無腦競爭。

到此這篇關(guān)于Java中的DelayQueue源碼解析的文章就介紹到這了,更多相關(guān)DelayQueue源碼解析內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++排序算法之桶排序原理及實(shí)現(xiàn)詳解

    C++排序算法之桶排序原理及實(shí)現(xiàn)詳解

    這篇文章主要介紹了C++排序算法之桶排序原理及實(shí)現(xiàn)詳解, C++ 桶排序是一種線性時(shí)間復(fù)雜度的排序算法,它通過將待排序元素分配到不同的桶中,然后對每個(gè)桶中的元素進(jìn)行排序,最后將所有桶中的元素按順序合并得到有序序列,需要的朋友可以參考下
    2023-10-10
  • Spring為何需要三級緩存解決循環(huán)依賴詳解

    Spring為何需要三級緩存解決循環(huán)依賴詳解

    這篇文章主要給大家介紹了關(guān)于Spring為何需要三級緩存解決循環(huán)依賴,而不是二級緩存的相關(guān)資料,這個(gè)也是一個(gè)Spring的高頻面試題,文中通過圖文介紹的非常詳細(xì),需要的朋友可以參考下
    2022-02-02
  • Spring整合quartz做定時(shí)任務(wù)的示例代碼

    Spring整合quartz做定時(shí)任務(wù)的示例代碼

    這篇文章主要介紹了在spring項(xiàng)目使用quartz做定時(shí)任務(wù),首先我這里的項(xiàng)目已經(jīng)是一個(gè)可以跑起來的完整項(xiàng)目,web.xml里面的配置我就不貼出來了,具體實(shí)例代碼跟隨小編一起看看吧
    2022-01-01
  • Maven中兩個(gè)命令clean 和 install的使用

    Maven中兩個(gè)命令clean 和 install的使用

    Maven是一個(gè)項(xiàng)目管理和自動(dòng)構(gòu)建工具,clean命令用于刪除項(xiàng)目中由先前構(gòu)建生成的target目錄,install命令用于將打包好的jar包安裝到本地倉庫中,供其他項(xiàng)目依賴使用,下面就來詳細(xì)的介紹一下這兩個(gè)命令
    2024-09-09
  • 詳解MyBatis配置typeAliases的方法

    詳解MyBatis配置typeAliases的方法

    這篇文章主要介紹了詳解MyBatis配置typeAliases的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • JAVA CountDownLatch(倒計(jì)時(shí)計(jì)數(shù)器)用法實(shí)例

    JAVA CountDownLatch(倒計(jì)時(shí)計(jì)數(shù)器)用法實(shí)例

    這篇文章主要介紹了JAVA CountDownLatch(倒計(jì)時(shí)計(jì)數(shù)器)用法實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-10-10
  • 基于JavaMail的Java郵件發(fā)送

    基于JavaMail的Java郵件發(fā)送

    電子郵件的應(yīng)用非常廣泛,例如在某網(wǎng)站注冊了一個(gè)賬戶,自動(dòng)發(fā)送一封歡迎郵件,通過郵件找回密碼,自動(dòng)批量發(fā)送活動(dòng)信息等。本文將簡單介紹如何通過 Java 代碼來創(chuàng)建電子郵件,并連接郵件服務(wù)器發(fā)送郵件
    2021-10-10
  • Spring中WebDataBinder使用詳解

    Spring中WebDataBinder使用詳解

    這篇文章主要為大家詳細(xì)介紹了Spring中WebDataBinder的使用,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-07-07
  • 詳解Java語言中一個(gè)字符占幾個(gè)字節(jié)?

    詳解Java語言中一個(gè)字符占幾個(gè)字節(jié)?

    這篇文章主要介紹了Java語言中一個(gè)字符占幾個(gè)字節(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • Java中的枚舉enum詳細(xì)解讀

    Java中的枚舉enum詳細(xì)解讀

    這篇文章主要介紹了Java中的枚舉enum詳細(xì)解讀,當(dāng)我們使用enum關(guān)鍵字開發(fā)一個(gè)枚舉類時(shí),默認(rèn)會(huì)繼承Enum類,而且是一個(gè)final類,當(dāng)有多個(gè)枚舉對象時(shí),使用逗號 ,隔開,最后一個(gè)用分號;結(jié)尾,需要的朋友可以參考下
    2024-01-01

最新評論