定时任务实现的关键DelayQueue延迟队列

之前学习定时任务线程池(ScheduledThreadPoolExecutor)时发现它主要依赖线程池和它的静态内部类DelayedWorkQueue实现。而DelayedWorkQueue就是一种延迟队列,今天学习是并发包提供的延迟队列(DelayQueue)。​

延迟队列说明

延迟队列提供的功能是在指定时间点才能获取队列元素的功能,队列最前面的元素是最优先执行的元素

列举一下使用场景可能能够更加好理解,比如缓存系统的设计,缓存中的对象,指定了过期时间,到了过期时间就需要从缓存中移出;在比如任务调度系统,要准确在任务规定的时间点执行任务。这些场景如果我们不使用延迟队列,就必须不同的遍历所有缓存、任务然后判断是否需要移除缓存、执行任务。

而延迟队列则不需要不停的扫描缓存、任务,它能够实现能够实现在准确的时间点去执行任务

接下来我们梳理一下缓存准时移除的实现,首先我们知道每个缓存的过期时间,就可以计算出每个缓存过期的时间戳,我们首先根据过期时间戳作为比较放到优先级队列(上一篇文章介绍的优先级队列)中,然后从优先级队列获取缓存,肯定是获取到最先需要过期的缓存,判断缓存是否到了过期时间,如果没到则把线程阻塞(过期时间戳与当前时间戳的差值),一定时间后线程自动唤醒,再次验证发现缓存刚好到期,可以移除缓存。

同样定时任务的准时执行也一样,只不过是把缓存过期的时间戳换成定时任务下次执行时间戳作为比较依据。

通过分析发现要实现这个功能需要一个优先级队列,保存的元素要指定移除队列的时间戳

DelayQueue属性介绍

从上一步分析得出延迟队列必须拥有优先级队列的功能,同时保存的元素要有确定移除队列的时间,那么我们来看DelayQueue具体实现,基本属性源码如下图:


可以看到它利用优先级队列q用来保存数据,那么就拥有了优先级队列的全部功能。

接着看它所能存储的元素都必须继承Delayed,查看Delayed源码发现它继承了Comparable接口,并且声明了方法“long getDelay(TimeUnit unit);”,这个方法的说明翻译过来是:以给定的时间单位返回与此对象关联的剩余延迟。它可以直接得出对象剩余延迟,就像缓存的剩余时间,就可以执行获取到对象的线程阻塞时间。

那么继承至Delayed的对象就同时拥有了优先级队列需要的Comparable的实现和对象剩余延迟执行的时间

DelayQueue关键实现

直接看take方法的实现,源码如下图:


只要上一篇文章弄懂了优先级队列,在看延迟队列的代码就很简单了。首先利用的是优先级队列获取元素,然后调用getDelay(Delayed接口声明的)方法判断是否阻塞以及阻塞时间

可以看到第一个线程进来了如果发现节点为null则是不限阻塞时间的阻塞,而后面进来的线程如果发现leader不为null也是直接阻塞,后面的线程可以通过leader线程执行完后唤醒,那么leader线程是什么时候唤醒呢?

leader无限阻塞的原因是队列中没有数据,所以要唤醒肯定就是添加数据的地方,在offer方法保存元素成功后会验证队列最前面的是不是刚刚保存的元素,如果是则会调用available.signal();唤醒线程,代码比较简单就不贴出来了。

延迟队列最佳实践

延迟队列要说最佳实践就要说到前面提交的定时任务了,在之前分析定时任务线程池ScheduledThreadPoolExecutor提到过,所以我们直接来看ScheduledThreadPoolExecutor中的实现,关键源码如下图:

每个任务在放到任务队列前都会设置下次执行时间time,如上图通过time实现了getDelay和compareTo方法,这样一个线程池就可以保存多个定时任务,每个任务在执行完后会重置time然后继续放到线程池的优先级队列中,就是如此的简单的实现了定时任务。

总结

在弄懂了优先级队列后在看延迟队列还是很简单的,无非是在优先级队列的基础上扩展了一丁点的功能,如何实现定时任务也就很简单了。

Java程序员日常学习笔记,如理解有误欢迎各位交流讨论!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352