所谓的延迟队列最大的特征是它可以自动通过队列进行脱离,例如:现在有一些对象被临时保存着,但是有可能该集合对象是一个公共对象,那么里面的某些数据如果不在使用的时候就希望其可以在指定的时间达到后自动的消失。

DelayQueue是延迟队列主要的使用类,所谓的延迟队列其实就是=BlockingQueue+PriorityQueue+Delayed

Delayed接口定义如下:

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

DelayQueue类定义如下:

public class DelayQueue<E extends Delayed> extends AbstractQueue<E> implements BlockingQueue<E>

延迟队列的基本使用

下面编写一个简单的延迟队列,延迟队列的本质就是到点后自动离开,那么就一定要一定要设置一个离开的时间点

范例:使用DelayQueue进行延迟队列的定义,以朋友聚会离开为例

package so.strong.mall.concurrent;

import java.util.concurrent.DelayQueue;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit; class Member implements Delayed {
private String name; //聚会人员的名字
private long expire; //失效时间,人员离开的时间,单位毫秒
private long delay; //设置延迟时间,单位毫秒(只呆多久) /**
* 设置参与队列之中的用户信息
*
* @param name 用户的额姓名
* @param delay 延迟时间
* @param unit 时间处理单位
*/
public Member(String name, long delay, TimeUnit unit) {
this.name = name;
this.delay = TimeUnit.MILLISECONDS.convert(delay, unit); //保存延迟的时间
this.expire = System.currentTimeMillis() + this.delay; //失效时间=当前时间加上延迟时间
} @Override
public long getDelay(TimeUnit unit) { //计算延迟是否到达
return unit.convert(this.expire - System.currentTimeMillis(), TimeUnit.MILLISECONDS);
} @Override
public int compareTo(Delayed o) { //决定优先级队列的弹出操作
return (int) (this.delay - this.getDelay(TimeUnit.MILLISECONDS));
} @Override
public String toString() {
return this.name + "预计" + TimeUnit.SECONDS.convert(this.delay, TimeUnit.MILLISECONDS) + "秒后离开,现在已经到点了";
}
} public class DelayDemo {
public static void main(String[] args) throws Exception {
System.out.println("*************准备聚会*************");
DelayQueue<Member> delayQueue = new DelayQueue<>();//设置延迟队列
delayQueue.add(new Member("Jack", 2, TimeUnit.SECONDS));
delayQueue.add(new Member("Rose", 4, TimeUnit.SECONDS));
while (!delayQueue.isEmpty()) { //如果聚会还有人
Delayed delayed = delayQueue.poll(); //从队列里面取出数据
System.out.println("[poll={" + delayed + "}]" + System.currentTimeMillis());
TimeUnit.MILLISECONDS.sleep(500); //延迟500毫秒
}
}
}
*************准备聚会*************
[poll={null}]1528094341989
[poll={null}]1528094342489
[poll={null}]1528094342990
[poll={null}]1528094343490
[poll={Jack预计2秒后离开,现在已经到点了}]1528094343991
[poll={null}]1528094344491
[poll={null}]1528094344991
[poll={null}]1528094345491
[poll={Rose预计4秒后离开,现在已经到点了}]1528094345992

使用延迟队列的最大特征就是想吃自助餐一样,给一个固定时间,只要到达了卓哥限制或者说自己定义的离开的时间,那么就可以离开。就可以使用poll()从队列里面把人给拽出来,如果不到时间不能拽。

延迟队列案例

下面以学生考试为例做一个说明,现在学生考试里面就可以有许多的学生参与考试,而且老师需要进行学生考试的监督,每一个学生考试的时间是不同的、,但是不管有多不同,一旦到点后一定要结束考试。本次利用学生的类概念和老师监督考试的概念来实现延迟队列的定义操作。

范例:教师监督学生考试模型

package so.strong.mall.concurrent;
import java.util.Random;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit; class Student implements Delayed {
private String name;
private long submitTime; //学生交卷时间,毫秒
private long workTime; //实际考试时间,毫秒 public Student(String name, long workTime, TimeUnit unit) {
this.name = name;
this.workTime = TimeUnit.MILLISECONDS.convert(workTime, unit);
this.submitTime = System.currentTimeMillis() + this.workTime;
} public void exam() { //考试处理
System.out.println("[" + this.name + "交卷-{" + this.submitTime + "}],交卷时间: "
+ System.currentTimeMillis() + ",花费时间:" + this.workTime);
} @Override
public long getDelay(TimeUnit unit) {
return unit.convert(this.submitTime - System.currentTimeMillis(), TimeUnit.MILLISECONDS);
} @Override
public int compareTo(Delayed o) {
return (int) (this.workTime - this.getDelay(TimeUnit.MILLISECONDS));
}
} class Teacher implements Runnable { //老师也是一个线程
private int studentCount = 0; //参与考试的学生数量
private int submitCount = 0; //保存交卷的学生个数
private DelayQueue<Student> students = null; public Teacher(DelayQueue<Student> students, int studentCount) {
this.students = students; //保存所有的学生信息
this.studentCount = studentCount; //保存学生数量
} @Override
public void run() {
System.out.println("************同学们开始答题*********");
try {
while (this.submitCount < this.studentCount) { //还有人未交卷
Student student = this.students.poll();
if (student != null) { //有人出队列了,表示有人交卷了
student.exam();//交卷处理
this.submitCount++; //交卷人数+1
}
}
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("************同学们结束考试*********");
}
} public class TestingDemo {
public static void main(String[] args) throws Exception{
Random random = new Random();// 利用随机数来模拟一个不同的交卷时间
DelayQueue<Student> students = new DelayQueue<>();
for (int i = 0; i < 10; i++) {
int time = random.nextInt(10);
while (time < 3) { //必须保证考试时间大于3秒
time = random.nextInt(10);
}
students.add(new Student("学生-" + i, time, TimeUnit.SECONDS));
}
new Thread(new Teacher(students, 10)).start();
}
}
************同学们开始答题*********
[学生-0交卷-{1528095504072}],交卷时间: 1528095504072,花费时间:8000
[学生-2交卷-{1528095501072}],交卷时间: 1528095504072,花费时间:5000
[学生-6交卷-{1528095500072}],交卷时间: 1528095504072,花费时间:4000
[学生-9交卷-{1528095499072}],交卷时间: 1528095504072,花费时间:3000
[学生-8交卷-{1528095505072}],交卷时间: 1528095505072,花费时间:9000
[学生-5交卷-{1528095499072}],交卷时间: 1528095505072,花费时间:3000
[学生-7交卷-{1528095501072}],交卷时间: 1528095505072,花费时间:5000
[学生-4交卷-{1528095504072}],交卷时间: 1528095505072,花费时间:8000
[学生-1交卷-{1528095500072}],交卷时间: 1528095505072,花费时间:4000
[学生-3交卷-{1528095500072}],交卷时间: 1528095505072,花费时间:4000
************同学们结束考试*********  

延迟队列到达时间点之后根据优先级进行队列的弹出处理。

延迟队列实现数据缓存

在延迟队列中有一个最大的特征就是到点后进行清除处理,那么在实际开发中,可以利用此机制来实现一个缓存的处理操作。正规的开发之中,往往都会利用数据层通过数据库进行该数据的取得,并且将这些数据变为VO的形式。

守护线程的作用:清楚队列中的数据,同时清楚map里面的数据。

范例:实现一个缓存的基础模型(以新闻缓存为例)

package so.strong.mall.concurrent;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit; class News { //定义一个新闻类
private String title;
private String note; public News(String title, String note) {
this.title = title;
this.note = note;
} @Override
public String toString() {
return "[新闻数据]title" + this.title + ",note=" + this.note;
}
} class DelayItem<T> implements Delayed {
private T item; //设置要保存的数据
private long expire; // 设置缓存的失效时间
private long delay; //设置要保存的时间 public DelayItem(T item, long delay, TimeUnit unit) {
this.item = item;
this.delay = TimeUnit.MILLISECONDS.convert(delay, unit);
this.expire = System.currentTimeMillis() + this.delay; } public T getItem() {
return item;
} @Override
public long getDelay(TimeUnit unit) {
return unit.convert(this.expire - System.currentTimeMillis(), TimeUnit.MILLISECONDS);
} @Override
public int compareTo(Delayed o) {
return (int) (this.delay - this.getDelay(TimeUnit.MILLISECONDS));
}
} //定义一个缓存的操作类,该类之中需要用户设置保存的可以Key类型与value类型
class Cache<K, V> {
//如果要实现多个线程的并发访问操作,必须考虑使用ConcurrentHashMap子类
private ConcurrentHashMap<K, V> cacheObjectMap = new ConcurrentHashMap<>();
private DelayQueue<DelayItem<Pair>> delayQueue = new DelayQueue<>(); private class Pair { //定义一个内部类,该类可以保存队列之中的K与V类型
private K key;
private V value; public Pair(K key, V value) {
this.key = key;
this.value = value;
}
} public Cache() {//如果要想清空不需要的缓存数据,则需要守护线程
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
while (true) { //守护线程要一直执行,当已经超时之后可以取出数据
DelayItem<Pair> item = Cache.this.delayQueue.poll();//通过延迟队列获取数据
if (item != null) { //已经有数据超时了
Pair pair = item.getItem();
Cache.this.cacheObjectMap.remove(pair.key, pair.value);
}
}
}
}, "缓存守护线程");
thread.setDaemon(true); //设置守护线程
thread.start(); //启动守护线程
} /**
* 表示将要保存的数据写入缓存中个,如果一个对象重复保存了,则应该重置它的超时时间
* @param key 要写入的k的内容
* @param value 要写入的对象
* @param time 保存的时间
* @param unit 保存的时间单位
*/
public void put(K key, V value, long time, TimeUnit unit) {
//put()方法如果发现原来的key存在,则会用新的value替换掉旧的内容,同时返回旧的内容
V oldValue = this.cacheObjectMap.put(key, value);
if (oldValue != null) { //原来已经存储过此数据
this.delayQueue.remove(key);
}
this.delayQueue.put(new DelayItem<>(new Pair(key, value), time, unit));
} public V get(K key) {
return this.cacheObjectMap.get(key);
}
} public class CacheDemo {
public static void main(String[] args) throws Exception {
Cache<String, News> cache = new Cache<>();
cache.put("aaa", new News("aaa", "xxx"), 3, TimeUnit.SECONDS);
cache.put("BBB", new News("BBB", "xxx"), 3, TimeUnit.SECONDS);
System.out.println(cache.get("aaa"));
System.out.println(cache.get("BBB"));
TimeUnit.SECONDS.sleep(5);
System.out.println(cache.get("aaa"));
System.out.println(cache.get("BBB"));
}
}
[新闻数据]titleaaa,note=xxx
[新闻数据]titleBBB,note=xxx
null
null  

缓存在实际开发之中是一个重要的开发组件,虽然在实际的工作之中会接触大量的缓存组件,但是其基本的实现原理可参考如上内容。

JUC——延迟队列的更多相关文章

  1. 如何用RabbitMQ实现延迟队列

    前言 在 jdk 的 juc 工具包中,提供了一种延迟队列 DelayQueue.延迟队列用处非常广泛,比如我们最常见的场景就是在网购或者外卖平台中发起一个订单,如果不付款,一般 15 分钟后就会被关 ...

  2. C#实现rabbitmq 延迟队列功能

    最近在研究rabbitmq,项目中有这样一个场景:在用户要支付订单的时候,如果超过30分钟未支付,会把订单关掉.当然我们可以做一个定时任务,每个一段时间来扫描未支付的订单,如果该订单超过支付时间就关闭 ...

  3. Java 延迟队列使用

    延时队列,第一他是个队列,所以具有对列功能第二就是延时,这就是延时对列,功能也就是将任务放在该延时对列中,只有到了延时时刻才能从该延时对列中获取任务否则获取不到…… 应用场景比较多,比如延时1分钟发短 ...

  4. Spring Boot(十四)RabbitMQ延迟队列

    一.前言 延迟队列的使用场景:1.未按时支付的订单,30分钟过期之后取消订单:2.给活跃度比较低的用户间隔N天之后推送消息,提高活跃度:3.过1分钟给新注册会员的用户,发送注册邮件等. 实现延迟队列的 ...

  5. 使用netty HashedWheelTimer构建简单延迟队列

    背景 最近项目中有个业务,需要对用户新增任务到期后进行业务处理.使用定时任务定时扫描过期时间,浪费资源,且不实时.只能使用延时队列处理. DelayQueue 第一想到的是java自带的延时队列del ...

  6. php使用redis的有序集合zset实现延迟队列

    延迟队列就是个带延迟功能的消息队列,相对于普通队列,它可以在指定时间消费掉消息. 延迟队列的应用场景: 1.新用户注册,10分钟后发送邮件或站内信. 2.用户下单后,30分钟未支付,订单自动作废. 我 ...

  7. C# RabbitMQ延迟队列功能实战项目演练

    一.需求背景 当用户在商城上进行下单支付,我们假设如果8小时没有进行支付,那么就后台自动对该笔交易的状态修改为订单关闭取消,同时给用户发送一份邮件提醒.那么我们应用程序如何实现这样的需求场景呢?在之前 ...

  8. rabbitmq延迟队列demo

    1. demo详解 1.1 工程结构: 1.2 pom 定义jar包依赖的版本.版本很重要,rabbit依赖spring,两者必须相一致,否则报错: <properties> <sp ...

  9. PHP 订单延时处理:延迟队列(未鉴定)

    PHP 订单延时处理:延迟队列: https://github.com/chenlinzhong/php-delayqueue

随机推荐

  1. [MongoDB]------windos下的安装部署与基础使用

    1.安装 首先前往官网进行下载,这里贴个地址https://www.mongodb.com/download-center#community 点击大大的原谅色的DOWNLOAD(msi)按钮进行下载 ...

  2. nginx alias

    A path to the file is constructed by merely adding a URI to the value of the root directive. If a UR ...

  3. Hadoop HA on Yarn——集群启动

    这里分两部分,第一部分是NameNode HA,第二部分是ResourceManager HA (ResourceManager HA是hadoop-2.4.1之后加上的) NameNode HA 1 ...

  4. 使用Thunderbird时你可能会用到的技巧

    1.添加qq邮箱账号 (1).开启IMAP/SMTP服务 先在QQ网页邮箱-设置-账户:开启IMAP/SMTP服务(2). Thunderbird 里设定端口(非POP):IMAP:imap.qq.c ...

  5. 2016-3-19日小结:scrollTop

    <div id="div1" style="padding: 0; position:absolute;width: 200px;height: 200px;< ...

  6. Day3JavaScript(一)JavaScript初识以及bom操作

    JavaScript简介 什么是JavaScript 弱类型,动态类型,基于原型的直译性的编程语言.1995年netscape(网景)在导航者浏览器中设计完成. JavaScript的特点 1.与HT ...

  7. Day8 类的继承

    为什么要继承? 观察两个类的成员组成 提取相同的属性和方法 宠物是父类,狗和金鱼是子类.子类具有父类的属性和方法. 继承定义 是使用已存在的类作为基础建立新类的技术. 单一继承:只有一个父类. 父类可 ...

  8. Python之括号()[]{}

    Python主要有三种数据类型:字典.列表.元组.其分别由花括号,中括号,小括号表示.如:字典:dic={'a':12,'b':34}列表:list=[1,2,3,4]元组:tup=(1,2,3,4)

  9. windows7使用Sphinx+PHP+MySQL详细介绍

    安装(Windows) 1.官方下载 Sphinx下载地址: 下载 2.解压并重命名 此处下载版本为3.0.3,将 sphinx 文件夹命名为sphinx 3.文件夹目录介绍 sphinx --api ...

  10. Mysql 安全登陆工具 mysql_config_editor

    mysql_config_editor 帮助信息请查看 man mysql_config_editor 或 mysql_config_editor -? 或 mysql_config_editor s ...