亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

《java.util.concurrent 包源碼閱讀》20 DelayQueue

2019-11-14 21:02:20
字體:
來源:轉載
供稿:網友
java.util.concurrent 包源碼閱讀》20 DelayQueue

DelayQueue有序存儲Delayed類型或者子類型的對象,沒當從隊列中取走元素時,需要等待延遲耗完才會返回該對象。

所謂Delayed類型,因為需要比較,所以繼承了Comparable接口:

public interface Delayed extends Comparable<Delayed> {    long getDelay(TimeUnit unit);}

其實Delayed對象的排序和延遲長短是無關的,因為Comparable的compare方法是用戶自己實現的,DelayQueue只是保證返回對象的延遲已經耗盡。

DelayQueue需要排序存儲Delayed類型的對象同時具備阻塞功能,但是阻塞的過程伴有延遲等待類型的阻塞,因此不能直接使用BlockingPRiorityQueue來實現,而是用非阻塞的版本的PriorityQueue來實現排序存儲。

private final PriorityQueue<E> q = new PriorityQueue<E>();

因此DelayQueue需要自己實現阻塞的功能(需要一個Condition):

private final Condition available = lock.newCondition();

老規矩還是先來看offer方法:

    public boolean offer(E e) {        final ReentrantLock lock = this.lock;        lock.lock();        try {            q.offer(e);            // 如果原來隊列為空,重置leader線程,通知available條件            if (q.peek() == e) {                leader = null;                available.signal();            }            return true;        } finally {            lock.unlock();        }    }

順便提一下,因為DelayQueue不限制長度,因此添加元素的時候不會因為隊列已滿產生阻塞,因此帶有超時的offer方法的超時設置是不起作用的:

    public boolean offer(E e, long timeout, TimeUnit unit) {        // 和不帶timeout的offer方法一樣        return offer(e);    }

因為DelayQueue需要自己實現阻塞,因此關注的重點應該是兩個帶有阻塞的方法:沒有超時的take方法和帶有超時的poll方法。

普通poll方法很簡單,如果延遲時間沒有耗盡的話,直接返回null就可以了。

    public E poll() {        final ReentrantLock lock = this.lock;        lock.lock();        try {            E first = q.peek();            if (first == null || first.getDelay(TimeUnit.NANOSECONDS) > 0)                return null;            else                return q.poll();        } finally {            lock.unlock();        }    }

接下來看take和帶timeout的poll方法,在看過DelayedWorkQueue之后這部分還是比較好理解的:

    public E take() throws InterruptedException {        final ReentrantLock lock = this.lock;        lock.lockInterruptibly();        try {            for (;;) {                // 如果隊列為空,需要等待available條件被通知                E first = q.peek();                if (first == null)                    available.await();                else {                    long delay = first.getDelay(TimeUnit.NANOSECONDS);                    // 如果延遲時間已到,直接返回第一個元素                    if (delay <= 0)                        return q.poll();                    // leader線程存在表示有其他線程在等待,那么當前線程肯定需要等待                    else if (leader != null)                        available.await();                    else {                        Thread thisThread = Thread.currentThread();                        leader = thisThread;                        // 如果沒有leader線程,設置當前線程為leader線程                        // 嘗試等待直到延遲時間耗盡(可能提前返回,那么下次                        // 循環會繼續處理)                        try {                            available.awaitNanos(delay);                        } finally {                            // 如果leader線程還是當前線程,重置它用于下一次循環。                            // 等待available條件時,鎖可能被其他線程占用從而導致                            // leader線程被改變,所以要檢查                            if (leader == thisThread)                                leader = null;                        }                    }                }            }        } finally {            // 如果沒有其他線程在等待,并且隊列不為空,通知available條件            if (leader == null && q.peek() != null)                available.signal();            lock.unlock();        }    }

再來看帶有timeout的poll方法,和DelayedWorkQueue非常相似:

    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                        // 嘗試等待available條件,記錄剩余的時間                        nanos = available.awaitNanos(nanos);                } else {                    long delay = first.getDelay(TimeUnit.NANOSECONDS);                    if (delay <= 0)                        return q.poll();                    if (nanos <= 0)                        return null;                    // 當leader線程不為空時(此時delay>=nanos),等待的時間                    // 似乎delay更合理,但是nanos也可以,因為排在當前線程前面的                    // 其他線程返回時會喚醒available條件從而返回,                    // 這里使用nanos和nonas<delay合并更加簡單                    if (nanos < delay || leader != null)                        nanos = available.awaitNanos(nanos);                    else {                        Thread thisThread = Thread.currentThread();                        leader = thisThread;                        try {                            long timeLeft = available.awaitNanos(delay);                            // nanos需要更新                            nanos -= delay - timeLeft;                        } finally {                            if (leader == thisThread)                                leader = null;                        }                    }                }            }        } finally {            if (leader == null && q.peek() != null)                available.signal();            lock.unlock();        }    }

前面理解了DelayedWorkQueue再來看DelayQueue就非常容易理解了。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲综合在线做性| 久久久av亚洲男天堂| 日韩欧美国产高清91| 欧美三级xxx| 日韩欧美在线第一页| 亚洲色图综合久久| 亚洲色图欧美制服丝袜另类第一页| 亚洲免费视频观看| 亚洲人成人99网站| 国产日韩欧美在线视频观看| 欧美巨猛xxxx猛交黑人97人| 亚洲色图第一页| 懂色av影视一区二区三区| 91色视频在线观看| 中文在线资源观看视频网站免费不卡| 亚洲欧美一区二区三区在线| 欧洲成人在线视频| 国产视频在线一区二区| 日韩电影在线观看永久视频免费网站| 日韩免费在线播放| 精品国产欧美一区二区三区成人| 国产欧美日韩免费看aⅴ视频| 97碰在线观看| 国产精品视频免费在线| 日韩在线一区二区三区免费视频| 日韩专区中文字幕| 中文字幕日韩欧美在线视频| 一本大道亚洲视频| 国产精品视频专区| 国产欧美一区二区三区在线看| 欧美午夜片欧美片在线观看| 精品亚洲va在线va天堂资源站| 欧美疯狂性受xxxxx另类| 久久亚洲精品国产亚洲老地址| 欧美亚州一区二区三区| 国产日韩欧美91| 91福利视频在线观看| 亚洲激情在线视频| 亚洲另类欧美自拍| 国产视频福利一区| 久久的精品视频| 亚洲自拍偷拍色图| 日韩av三级在线观看| 欧美黑人xxxx| 8x海外华人永久免费日韩内陆视频| 亚洲成人三级在线| 91精品视频在线免费观看| 国产成人av在线| 日韩中文字在线| 亚洲欧美日韩国产成人| 中文字幕日韩综合av| 亚洲最大福利网| 亚洲最大福利视频| 国产精品丝袜高跟| 久久久精品2019中文字幕神马| 国产精品吹潮在线观看| 日韩中文字幕国产精品| 欧美性视频精品| 亚洲影视中文字幕| 国产精品久久久久久久9999| 精品久久久久久久久久久久久| 午夜精品在线视频| 庆余年2免费日韩剧观看大牛| 国产成人亚洲综合青青| 亚洲欧美在线x视频| 欧美大片在线影院| 成人精品久久一区二区三区| 97高清免费视频| 福利一区视频在线观看| 91精品国产综合久久久久久久久| 精品亚洲一区二区三区在线观看| 日韩亚洲精品视频| 亚洲欧美在线x视频| 亚州国产精品久久久| 欧美一级免费看| 成人午夜在线视频一区| 成人免费激情视频| 欧美另类69精品久久久久9999| 国产日韩欧美另类| 欧美丝袜一区二区| 亚洲国产91色在线| 97不卡在线视频| 国内揄拍国内精品少妇国语| 91精品在线播放| 黑人巨大精品欧美一区二区| 最近2019年好看中文字幕视频| 55夜色66夜色国产精品视频| 欧美巨大黑人极品精男| 日韩精品久久久久久福利| 91久久精品美女高潮| 黑人精品xxx一区一二区| 亚洲第一区第一页| 日韩电影在线观看永久视频免费网站| 亚洲社区在线观看| 国产在线精品成人一区二区三区| 亚洲精品第一页| 欧美成人一区二区三区电影| 欧美一级淫片videoshd| 亚洲国产婷婷香蕉久久久久久| 精品欧美国产一区二区三区| 精品中文视频在线| 欧美—级a级欧美特级ar全黄| 一区二区在线视频播放| 日韩极品精品视频免费观看| 色噜噜久久综合伊人一本| 国产精品久久久久久久久久尿| 国产欧美日韩91| 日本三级韩国三级久久| 红桃视频成人在线观看| 久久精视频免费在线久久完整在线看| 中文字幕日韩欧美在线视频| 97国产在线观看| 欧美有码在线视频| 国产成人小视频在线观看| 国产一区二区三区在线播放免费观看| 91精品国产自产91精品| 97国产suv精品一区二区62| 久久人人爽人人| 992tv成人免费视频| 欧美电影院免费观看| 亚洲国产97在线精品一区| 欧美视频在线看| 成人激情免费在线| 成人午夜一级二级三级| 久久综合88中文色鬼| 亚洲伊人第一页| 日韩中文字幕不卡视频| 欧美成人精品不卡视频在线观看| 久久久女女女女999久久| 亚洲欧美综合图区| 懂色av中文一区二区三区天美| 亚洲国产精品一区二区久| 欧美成在线视频| 亚洲自拍偷拍区| 久久精品视频播放| 中文一区二区视频| 97精品一区二区三区| 国产精品69精品一区二区三区| 亚洲视频在线观看视频| 欧美精品电影免费在线观看| 91在线无精精品一区二区| 国产精品国产三级国产专播精品人| 色黄久久久久久| 欧美成人剧情片在线观看| 亚洲mm色国产网站| 日本精品视频网站| 久久久av亚洲男天堂| 久热精品视频在线免费观看| 国外成人在线视频| 国产精品视频内| 日韩电影免费在线观看| 亚洲成人网av| 欧美成人一二三| 久久久久久午夜| 国产精品久久av| 亚洲成人免费网站| 青青草精品毛片| 日韩在线国产精品| 91精品在线观看视频| 91免费综合在线| 国语自产精品视频在线看抢先版图片| 亚洲а∨天堂久久精品9966| 亚洲第一级黄色片| 日韩大片在线观看视频|