DelayQueue原理分析

news2025/1/9 1:10:28

1.简介

DelayQueue同样也是适用于并发环境下的容器之一,该容器属于阻塞队列的一种,其底层数据结构是PriorityQueue,主要应用于执行定时任务和缓存过期删除的场景。
DelayQueue也是线程安全的,它通过内部的ReentrantLock实现了线程间的互斥访问。
DelayQueue要求其内部元素必须实现Delayed接口,并重写getDelay方法。默认情况下,DelayQueue会按照元素的到期时间进行升序排列,且仅当元素到期(getDelay() <= 0)时,才能取出该元素。

2.实现原理

  • 为了实现延时的语义,其内部采用PriorityQueue进行延时任务的存储与管理,通过强制每个元素实现Delayed接口以实现getDelaycompareTo方法,从而实现了延时低的任务优先被执行的目标。
  • 内部设置了一把ReentrantLock,从而实现了进程间的互斥访问,线程安全性得到保证。
  • 内部还配有Condition,通过awaitsignal方法来完成多线程间的唤醒与等待。

3.源码分析

3.1 类定义

DelayQueue的类定义(JDK17)")
DelayQueue继承自AbstractQueue,因此具有一些基本的队列的增删改查的模板操作;
同时,它还实现了BlockQueue接口,因此该队列将具备阻塞队列的一些性质。
此外,我们还注意到,DelayQueue的泛型参数必须为Delayed接口的子类,这也印证了我们上面所说的一点:DelayQueue的元素必须强制实现Delayed接口。
同时,Delayed接口又继承了Comparable接口,因此,DelayQueue中的元素天然具有可排序的特性。

Q:为什么要强制队列元素实现Delayed接口?
A
实现Delayed接口,具体来讲是实现其要求的getDelaycompareTo这两个方法。

  • 实现getDelay方法:用于指示延时任务的剩余延迟时间,作为执行线程衡量该延时任务目前能不能够被执行的标准。
  • 实现compareTo方法:用于作为内部PriorityQueue的排序标准,用于比较任务的优先级,以编排好当前延时任务队列的正确的执行顺序。

DelayQueue的继承实现链(JDK17)")

3.2 初始化

DelayQueue的构造方法源码(JDK17)")
构造方法逻辑很简单,对于有参构造函数,它将调用addAll方法,而在addAll中,待要添加的集合校验通过后,遍历整个集合并挨个儿调用add方法,而在add方法中又调用了offer方法,这个我们待会细说,最终如果一切没问题的话,那么modified将被置为true并返回,否则,只要有一个元素添加失败就会抛异常(往往是由于队列满导致的)。
调用链:有参构造 --> addAll --> 遍历调用add --> offer

3.3 关键字段说明

DelayQueue类中的部分关键字段(JDK17)")

  • qDelayQueue的核心,也是其底层数据结构,用于存放延时任务,后续任务的添加与执行都是要靠它来完成的。同时它作为一个优先级队列,会将任务按照其延时时间(通过重写Delayed接口的父接口Comparable中的compareTo方法)进行升序排序。
  • leader:执行延时任务的唯一线程,以防止多个线程争抢而使任务执行效率降低。(相当于领导者-追随者模式中的领导者)
  • available:唤醒因到来的时候队列为空而等待或者到来时已经有其他线程在处理任务而等待的线程或者满足这些条件而阻塞,实现了工作线程间的阻塞与唤醒。

3.4 添加

DelayQueue的部分添加(生产)元素的方法(JDK17)元素的方法(JDK17)")
你会发现,不论是add方法还是put方法,都是直接调用的offer方法,因此,这里我们只分析offer执行流程

  1. 加锁。
  2. 调用内部PriorityQueue上的offer方法添加元素。
  3. 判断添加了该元素后的PriorityQueue中的头部元素是不是当前添加进入的元素(即当前所添加的元素是不是延时最短的那个),若是,则置leader线程为null,同时唤醒阻塞在available条件上的线程;否则,不做任何处理。
  4. 释放锁。

3.5 删除(执行延时任务)

DelayQueue的执行延时任务的方法(JDK17)")
**poll**方法的执行流程

  1. 加锁。
  2. 调用PriorityQueuepeek方法,试探性的取一下队头元素。
  3. 若发现队头元素(延时任务)为null或者还没到达预定的延时时间,则返回null;否则,调用poll方法将该元素出队并返回。
  4. 释放锁。

**take**方法的执行流程

  1. 加锁。
  2. 试探性的取到队列的头部元素:
    1. 如果头部元素为null,则说明队列目前没有延时任务可供消费,因此阻塞在available条件上。
    2. 若头部元素不为null
      1. 如果该元素已到达或者超过延时时间,则调用poll方法出队并返回。
      2. 若还未到达延时时间,则查看一下当前领导者线程是否为空:
        1. 若不为空,则说明目前已经有线程正在处理延时任务,因此我们需要阻塞等待,因此在available条件上阻塞。
        2. 若为空,则令当前线程作为领导者线程,然后阻塞等待至预定的延时执行时间,等待一段时间后,再次将主线程置为null,然后进入到for循环的下一次循环,进入到b -> i分支,处理延时任务(这也是为什么设置for循环的原因)。
  3. 最后的最后,如果领导者线程为null并且队列中还有延时任务,则随机唤醒阻塞在available条件上的线程以进行下一个延时任务的等待&处理。
  4. 释放锁。

上述过程其实用到了一个设计模式:领导者-追随者模式。

通过过程分析,不难发现,poll是非阻塞式删除,take是阻塞式删除。

3.6 获取

DelayQueue的获取但不移除元素的方法(JDK17)")
执行流程

  1. 加锁。
  2. 调用PriorityQueuepeek方法,然后直接返回该元素。
  3. 释放锁。

3.7 获取元素数量

DelayQueue的获取元素数量的方法(JDK17)")
执行流程

  1. 加锁。
  2. 调用PriorityQueuesize方法并返回。
  3. 释放锁。

4.评估

如果线程数量过少,且处理的任务耗时较长,而后续延时任务的到期时间相对集中,那么可能会使得后面的延时任务出现延期处理的情况。
线程数量过多也不见得是一件好事,因为这将在线程调度、同步方面花费更多的时间。
(当然,DelayQueue的工作线程的数量固定为1,上面是针对[领导者-追随者模式](https://www.yuque.com/jujingyi-mzjzr/ybr4gh/ix3ifmtyb3rfllxf) + DelayQueue而言的)

参考文档

DelayQueue 源码分析
Java 延迟队列 DelayQueue 的原理

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1688179.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

error1310 写入文件时发生错误,请确认您是否有访问权限 也可能出现error 1304 :写入文件时出错

一般错误提示如下 error1310 Error writing to file 错误 1310 &#xff1a;写入文件时出错&#xff1a;请确认您有权访问该目录&#xff0c; error1304 Error writing to file 错误 1304 &#xff1a;写入文件时出错&#xff1a;请确认您有权访问该目录 1.首先我们退出所…

【保姆级教程】基于OpenCV+Python的人脸识别上课签到系统

【保姆级教程】基于OpenCVPython的人脸识别上课签到系统 一、软件安装及环境配置1. 安装IDE&#xff1a;PyCharm2. 搭建Python的环境3. 新建项目、安装插件、库 二、源文件编写1. 采集人脸.py2. 训练模型.py3. 生成表格.py4. 识别签到.py5. 创建图形界面.py 三、相关函数分析1.…

【文心智能体】创建一个属于自己的生活情感类智能体

文章目录 前言一、创建智能体二、体验 前言 智能体技术的快速发展&#xff0c;进一步激发了各行业开发者对其实际应用及用户需求的深入探索。 创建一个属于自己的智能体。文心一言提供了一个很好的平台。 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考…

【运维】笔记本电脑风扇清洁

笔记本电脑是我们不可或缺的工具&#xff0c;无论是工作、学习还是娱乐。然而&#xff0c;随着时间的推移&#xff0c;笔记本电脑的性能可能会因为各种原因受到影响&#xff0c;尤其是散热问题。过热不仅会降低性能&#xff0c;还可能缩短硬件的寿命。最近&#xff0c;在使用我…

CATIA入门操作——萌新宝宝遇到的奇奇怪怪的问题解决,持续更新中。。。

目录 引出发生肾么事了&#xff1f;&#xff1f;鼠标中键旋转不了解决&#xff1a;特征树不显示参数关系 我的窗口去哪了&#xff1f;插曲&#xff1a;草图工具的调出插曲&#xff1a;颜色工具栏显示 弹窗警告警告&#xff1a;创建约束是临时的 操作技巧技巧&#xff1a;快速隐…

ROS | C++和python实现IMU数据获取

实验步骤&#xff1a; 协方差矩阵的用途&#xff1a; C: Pyhton:

网络安全、信息安全、数据安全的定义与区别

信息安全 信息安全是指信息的保密性、完整性、可用性和真实性的保持。从定义角度来说&#xff0c;信息安全没有严格标准定义&#xff0c;但从信息安全涉及的内容出发&#xff0c;信息安全确保信息存储或传输中的信息&#xff0c;不被他人有意或无意的窃取与破坏。这里的“信息”…

hive初始化失败报错:Error: Duplicate key name ‘PCS_STATS_IDX‘ (state=42000,code=1061)

意思是key name ‘PCS_STATS_IDX’ (state42000,code1061)重复了&#xff0c;问题出在不是第一次初始化&#xff0c;因为我们在hive-site.xml中配置了 javax.jdo.option.ConnectionURL jdbc:mysql://192.168.200.137:3306/metastore?createDatabaseIfNotExisttrue JDBC conne…

PE文件入门,一篇就够了

title: PE文件学习篇 tags: - Windows - PE 最近在准备面试相的内容&#xff0c;对pe相关的问题有些生疏了&#xff0c;于是就边看博客复习边整理到论坛上希望对大家有帮助。 在《逆向工程核心原理》这本书接触到了PE文件&#xff0c;但是当时学不进去&#xff0c;感觉很晦涩…

pytorch在docker里面使用GPU

本博客主要介绍如何在容器里面使用pytorch进行推理&#xff0c;训练&#xff0c;同时用上GPU。 1. 前置条件&#xff0c;安装好docker。 2. 安装nvidia-container-toolkit 参考官方文档&#xff1a; Installing the NVIDIA Container Toolkit — NVIDIA Container Toolkit 1.…

【汽车操作系统】Autosar和商用OS

目录 什么是AUTOSAR? CP AUTOSAR架构 CAN通信 AP AUTOSAR 背景 CP&AP 开发方面的不同&#xff1a; WRLinux介绍 QNX介绍 什么是AUTOSAR? 随着汽车功能越来越多&#xff0c;导致ECU的数量越来越多。1993年的时候&#xff0c;奥迪A8才只有5个ECU现在典型的现代汽车…

SSRF服务端请求伪造漏洞原理与修复及靶场实践

SSRF服务端请求伪造漏洞原理与修复及靶场实践 SSRF漏洞原理与检测 SSRF&#xff08;Server-Side Request Forgery&#xff0c;服务器端请求伪造&#xff09;漏洞是一种因为服务端提供了远程访问服务&#xff0c;而并未对请求目标进行限制或限制不严格而引起的安全漏洞&#x…

C语言/数据结构——每日一题(有效的括号)

一.前言 如果想要使用C语言来解决这道题——有效的括号&#xff1a;https://leetcode.cn/problems/valid-parentheses/description/我们必须要借用上一篇我们所讲的内容——栈的实现&#xff1a;https://blog.csdn.net/yiqingaa/article/details/138923750?spm1001.2014.3001.…

leetcode124 二叉树中的最大路径和-dp

题目 二叉树中的 路径 被定义为一条节点序列&#xff0c;序列中每对相邻节点之间都存在一条边。同一个节点在一条路径序列中 至多出现一次 。该路径 至少包含一个 节点&#xff0c;且不一定经过根节点。 路径和 是路径中各节点值的总和。 给你一个二叉树的根节点 root &…

SpringCloud微服务04-Elasticsearch-DSL查询-聚合

一、Elasticsearch 搜索引擎&#xff0c;响应速度非常快&#xff0c;特别是对大数据量的情况 1.初始elasticsearch 如果只需要商品搜索&#xff0c;百度这种搜索网站&#xff0c;只需要第二个就够了 docker部署&#xff1a;day08-Elasticsearch - 飞书云文档 (feishu.cn) e…

我爱我家:租赁下位替代买房,能行吗?

我爱我家&#xff0c;凭什么五天四板&#xff1f; 上周五的楼市组合拳出台后&#xff0c;地产板块迎来高潮。 这其中最火的不是我们常说的“招宝万金”&#xff0c;而是——我爱我家。 五天四板&#xff0c;一个月不到&#xff0c;股价轻松翻翻。 公司有什么变化吗&#xff1…

蓝桥杯嵌入式国赛笔记(2):拓展板按键程序设计

目录 1、前言 2、电路原理 3、代码编写 3.1 读取Btn电压 3.2 检索按键 3.3 main文件编写 3.3.1 进行变量定义 3.3.2 AD_Key函数 3.3.3 LCD函数 3.3.4 main函数 3.3.5 完整代码 4、测试 5、总结 1、前言 本文进行拓展板按键程序设计&#xff0c;拓展板的按键是通…

分布式事务解决方案(强一致性)

强一致性事务概述 分布式事务领域&#xff0c;最早采用的是符合CAP理论的强一致性事务方案来解决分布式事务问题&#xff0c;强一致性分布式事务要求在任意时刻查询参与全局事务的各个节点的数据都是一致的 典型案例&#xff1a; 包括DTP模型&#xff08;全局事务模型&#x…

【Crypto】MD5

文章目录 MD5解题感悟 MD5 提示的很明显MD5 小小flag&#xff0c;拿下&#xff01; 解题感悟 没啥感悟…

LLM 大模型学习必知必会系列(十一):大模型自动评估理论和实战以及大模型评估框架详解

LLM 大模型学习必知必会系列(十一)&#xff1a;大模型自动评估理论和实战以及大模型评估框架详解 0.前言 大语言模型&#xff08;LLM&#xff09;评测是LLM开发和应用中的关键环节。目前评测方法可以分为人工评测和自动评测&#xff0c;其中&#xff0c;自动评测技术相比人工…