简介

今天给大家介绍一下DelayQueue,DelayQueue是BlockingQueue的一种,所以它是线程安全的,DelayQueue的特点就是插入Queue中的数据可以按照自定义的delay时间进行排序。只有delay时间小于0的元素才能够被取出。

DelayQueue

先看一下DelayQueue的定义:

public class DelayQueue extends AbstractQueue

implements BlockingQueue

从定义可以看到,DelayQueue中存入的对象都必须是Delayed的子类。

Delayed继承自Comparable,并且需要实现一个getDelay的方法。

为什么这样设计呢?

因为DelayQueue的底层存储是一个PriorityQueue,在之前的文章中我们讲过了,PriorityQueue是一个可排序的Queue,其中的元素必须实现Comparable方法。而getDelay方法则用来判断排序后的元素是否可以从Queue中取出。

DelayQueue的应用

DelayQueue一般用于生产者消费者模式,我们下面举一个具体的例子。

首先要使用DelayQueue,必须自定义一个Delayed对象:

@Data

public class DelayedUser implements Delayed {

private String name;

private long avaibleTime;

public DelayedUser(String name, long delayTime){

this.name=name;

//avaibleTime = 当前时间+ delayTime

this.avaibleTime=delayTime + System.currentTimeMillis();

}

@Override

public long getDelay(TimeUnit unit) {

//判断avaibleTime是否大于当前系统时间,并将结果转换成MILLISECONDS

long diffTime= avaibleTime- System.currentTimeMillis();

return unit.convert(diffTime,TimeUnit.MILLISECONDS);

}

@Override

public int compareTo(Delayed o) {

//compareTo用在DelayedUser的排序

return (int)(this.avaibleTime - ((DelayedUser) o).getAvaibleTime());

}

}

上面的对象中,我们需要实现getDelay和compareTo方法。

接下来我们创建一个生产者:

@Slf4j

@Data

@AllArgsConstructor

class DelayedQueueProducer implements Runnable {

private DelayQueue delayQueue;

private Integer messageCount;

private long delayedTime;

@Override

public void run() {

for (int i = 0; i < messageCount; i++) {

try {

DelayedUser delayedUser = new DelayedUser(

new Random().nextInt(1000)+"", delayedTime);

log.info("put delayedUser {}",delayedUser);

delayQueue.put(delayedUser);

Thread.sleep(500);

} catch (InterruptedException e) {

log.error(e.getMessage(),e);

}

}

}

}

在生产者中,我们每隔0.5秒创建一个新的DelayedUser对象,并入Queue。

再创建一个消费者:

@Slf4j

@Data

@AllArgsConstructor

public class DelayedQueueConsumer implements Runnable {

private DelayQueue delayQueue;

private int messageCount;

@Override

public void run() {

for (int i = 0; i < messageCount; i++) {

try {

DelayedUser element = delayQueue.take();

log.info("take {}",element );

} catch (InterruptedException e) {

log.error(e.getMessage(),e);

}

}

}

}

在消费者中,我们循环从queue中获取对象。

最后看一个调用的例子:

@Test

public void useDelayedQueue() throws InterruptedException {

ExecutorService executor = Executors.newFixedThreadPool(2);

DelayQueue queue = new DelayQueue<>();

int messageCount = 2;

long delayTime = 500;

DelayedQueueConsumer consumer = new DelayedQueueConsumer(

queue, messageCount);

DelayedQueueProducer producer = new DelayedQueueProducer(

queue, messageCount, delayTime);

// when

executor.submit(producer);

executor.submit(consumer);

// then

executor.awaitTermination(5, TimeUnit.SECONDS);

executor.shutdown();

}

上面的测试例子中,我们定义了两个线程的线程池,生产者产生两条消息,delayTime设置为0.5秒,也就是说0.5秒之后,插入的对象能够被获取到。

线程池在5秒之后会被关闭。

运行看下结果:

[pool-1-thread-1] INFO com.flydean.DelayedQueueProducer - put delayedUser DelayedUser(name=917, avaibleTime=1587623188389)

[pool-1-thread-2] INFO com.flydean.DelayedQueueConsumer - take DelayedUser(name=917, avaibleTime=1587623188389)

[pool-1-thread-1] INFO com.flydean.DelayedQueueProducer - put delayedUser DelayedUser(name=487, avaibleTime=1587623188899)

[pool-1-thread-2] INFO com.flydean.DelayedQueueConsumer - take DelayedUser(name=487, avaibleTime=1587623188899)

我们看到消息的put和take是交替进行的,符合我们的预期。

如果我们做下修改,将delayTime修改为50000,那么在线程池关闭之前插入的元素是不会过期的,也就是说消费者是无法获取到结果的。

总结

DelayQueue是一种有奇怪特性的BlockingQueue,可以在需要的时候使用。

以上就是详解java中DelayQueue的使用的详细内容,更多关于java DelayQueue的资料请关注脚本之家其它相关文章!

java delayqueue_详解java中DelayQueue的使用相关推荐

  1. 【java】详解Java的类文件(class文件)结构

    1.概述 转载:详解Java的类文件(class文件)结构 大家好,我是二哥呀,今天我拿了一把小刀,准备解剖一下 Java 的 class 文件. CS 的世界里流行着这么一句话,"计算机科 ...

  2. java deepcopy_详解JAVA 深层拷贝 DeepCopy的使用方式

    说到复制方法,在开发程序中要怎么复制呢?其实复制Java数组的方法很多,但大多数都是浅层复制,今天爱站技术频道小编带你寻找详解JAVA 深层拷贝 DeepCopy的使用方式. 方法实现很简单,提供两种 ...

  3. java list详解_Java 中 list 用法案例详解

    LIST是个容器接口可以理解为动态数组,传统数组必须定义好数组的个数才可以使用,而容器对象无须定义好数组下标总数,用add()方法即可添加新的成员对象,他可以添加的仅仅只能为对象,不能添加基本数据类型 ...

  4. java构造器详解_Java中关于构造器的使用详解

    这篇文章主要介绍了Java构造器使用方法及注意事项的相关资料,这里举例说明如何使用构造器及需要注意的地方,需要的朋友可以参考下 Java构造器使用方法及注意事项 超类的构造器在子类的构造器运行之前运行 ...

  5. 【java】详解java中的注解(Annotation)

    目录结构: contents structure [+] 什么是注解 为什么要使用注解 基本语法 4种基本元注解 重复注解 使用注解 运行时处理的注解 编译时处理的注解 1.什么是注解 用一个词就可以 ...

  6. java toast_详解Android中的Toast源码

    Toast源码实现 Toast入口    我们在应用中使用Toast提示的时候,一般都是一行简单的代码调用,如下所示: [java] view plaincopyprint?在CODE上查看代码片派生 ...

  7. java optional详解_java8中Optional的用法详解

    package com.lyzx.concurrent.java8; import org.junit.Test; import java.util.Optional; import java.uti ...

  8. java dateutils详解_java中DateUtils时间工具类详解

    package com.example.administrator.myapp; import java.text.ParseException; import java.text.SimpleDat ...

  9. java命令详解 java -D

    2019独角兽企业重金招聘Python工程师标准>>> JAVA 命令参数详解: 1.-D<name>=<value> set a system proper ...

  10. java nio详解,Java NIO API详解

    Java NIO API详解 在JDK 1.4以前,Java的IO操作集中在java.io这个包中,是基于流的阻塞(blocking)API.对于大多数应用来说,这样的API使用很方 便,然而,一些对 ...

最新文章

  1. 解析nginx负载均衡
  2. 阿尔法狗要逆天!韩专家称其故意输李世石一局
  3. 《中国人工智能学会通讯》——8.25 基于演化优化的生物网络配准
  4. 使类和成员的可访问性最小化
  5. 现代软件工程 10 绩效管理
  6. python初学小游戏_python学习(小游戏)
  7. Zabbix 配置钉钉脚本告警(4)
  8. 怎样编写和执行Js文件
  9. 数据库——“该连接已关闭”问题解决方案
  10. 《AIX 5L 系统管理技术》学习笔记之第七章设备管理
  11. java高级工程师 考试_JAVA高级工程师笔试题及答案
  12. Win10鼠标宏怎么设置?电脑设置鼠标宏的方法
  13. roboware studio安装
  14. 计算机联锁设备的操作,计算机联锁系统设备的键盘操作
  15. 今日头条笔试题 数列
  16. eclipse下载速度过慢的解决方法
  17. 生活时尚酒店品牌JOJOE进军中国,首批选址7个城市,未来将开设至少1300家门店 | 美通社头条...
  18. AI内容生成工具列表,可用于生成 AI 内容的工具列表。具有图像生成、文本生成等功能。
  19. rtx3080ti和3090差距 rtx3080ti和3090哪个性价比高
  20. 只有你能听见(Calling you)2

热门文章

  1. 3种常见软件项目工作量评估方法简述
  2. 数据总线和地址总线区别是什么?作用分别是什么
  3. Speex的安装使用
  4. Linux数据库迁移
  5. LabVIEW开放神经网络交互工具包【ONNX】,大幅降低人工智能开发门槛,实现飞速推理
  6. 学习数码相框1.2.0.0字符的编码方式_显示点阵文字_freetype_在PC上测试freetype
  7. 软件系统设计-12-软件设计体系结构概述
  8. openwrt折腾记2-广告拦截adbyby与pass
  9. 【LKA】国内车道相关数据
  10. OSI网络七层协议与TCPIP协议