java延迟队列
大多数用到定时执行的功能都是用任务调度来做的,单身当碰到类似订餐业务/购物等这种业务就不好处理了,比如购物的订单功能,在你的订单管理中有N个订单,当订单超过十分钟未支付的时候自动释放购物车中的商品,订单失效。这种高频率的延迟任务再用任务调度(定时)实现就得不偿失了。推荐用Java延迟队列来实现,DelayQueue是java.util.concurrent中提供的一个类DelayQueue是一个无界的BlockingQueue,用于放置实现了Delayed接口的对象,其中的对象只能在其到期时才能从队列中中取走。这种队列是有序的,即对头对象的延迟到期时间最长。注意:不能将null元素放置到这种队列中。
1、java延迟队列实现方式
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong; //任务线程 实现delayed接口
public class DelayItem<T extends Runnable> implements Delayed {
//到期时间
private final long time;
//任务对象
private final T task;
//原子类
private static final AtomicLong atomic = new AtomicLong(0);
private final long n; public DelayItem(long timeout, T t) {
this.time = System.nanoTime() + timeout;
this.task = t;
this.n = atomic.getAndIncrement();
}
//返回与此对象相关的剩余延迟时间,以给定的时间单位表示
public long getDelay(TimeUnit unit) {
return unit.convert(this.time - System.nanoTime(),TimeUnit.NANOSECONDS);
} public int compareTo(Delayed other) {
if(other == this) {
return 0;
}
if(other instanceof DelayItem) {
DelayItem<?> x = (DelayItem<?>)other;
long diff = tiem - x.time;
if(diff < 0) {
return -1;
}else if(diff > 0) {
return 1;
}else if( n < x.n){
return -1;
}else {
return 1;
}
long d = (getDelay(TimeUnit.NANOSECONDS) - other.getDelay(TimeUnit.NANOSECONDS));
return (d == 0) ? 0 : ((d < 0) ? -1 : 1);
}
public T getTask(){
return this.task;
}
@Override
public int hashCode(){
return task.hashCode();
}
@Override
public boolean equals(Object object){
if(object instanceof DelayItem){
return object.hashCode() == hashCode() ? true : false;
}
return false;
}
} //管理延迟任务的类
import java.util.Map;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import org.apache.log4j.Logger;
//延迟队列存放有效期对象
public class ItemQueueThread {
private static final Logger logger = Logger.getLogger(this.class);
private ItemQueueThread(){}
//延迟加载(线程安全)
private static class LazyHolder{
private static ItemQueueThread itemQueueThread = new ItemQueueThread();
}
public static ItemQueueThread getInstance(){
return LazyHolder.itemQueueThread;
}
//缓存线程池
ExecutorService executor = Executors.newCacheThreadPool();
//线程
private Thread daemonThead;
//初始化线程
public void init() {
daemonThread = new Thread(() -> {
try{
execute();
}cathc(InterruptedException e){
e.printStackTrace();
logger.info(e.getMessage());
}
});
System.out.println("init......start");
daemonThread.start();
} private void execute() throws InterrupedException {
while(true) {
Map<Thread, StackTraceElement[]> map = Thread.getAllStackTraces();
System.out.println("线程数...." + map.size());
System.out.println(System.currentTimeMills());
System.out.println(item.size());
System.out.println("线程状态----" + Thread.currentThread().getState());
try{
//从延迟队列中取值,如果没有对象过期责队列一直等待
DelayItem<?> t1 = item.take();
if(t1 != null){
Runnable task = t1.getTask();
if(task == null){
continue;
}
executor.execute(task);
}
}catch(Exception e) {
e.printStackTrace();
logger.info(e.getMessage());
}
}
}
//创建空的延迟队列
private DelayQueue<DelayItem<?>> item = new DelayQueue<>();
//往队列中添加任务
public void put(long time, Runnable task, TimeUnit timeUnit){
//转换成ns
long nanoTime = TimeUnit.NANOSECONDS.convert(time,timeUnit);
DelayItem<?> k = new DelayItem(nanoTime,task);
item.put(k);_:
}
//结束任务
public boolean endTask(DelayItem<Runnable> task){
return item.remove(task);
}
} //把需要延迟的功能代码单独抽取出来作为一个类,继承Runnable实现run方法
public class DataDemo implements Runnable {
int a = -1;
public DataDemo(int i){
this.a = i;
}
@Override
public void run(){
System.out.println("超时,要撤销订单...." + a);
}
} //test class
import java.util.Random;
import java.util.concurrent.TimeUnit;
public class DelayTest{
public static void main(String[] args){
ItemQueueThread ith = ItemQueueThread.getInstance();
ith.init();
Random r = new Random();
for(int i = 0; i < 5; i++){
int a = r.nextInt(20);
System.out.println("预先知道等待时间:" + a);
DataDemo dd = new DataDemo(a);//创建一个任务对象
ith.put(a,dd,TimeUnit.SECONDS);//将任务添加到队列中
}
}
}
//注意ItemQueueThread的init方法,要在容器初始化的时候就要执行,或在第一次put延迟对象任务之前就要初始化完成,当设定的延迟时间到期时会执行任务对象中的run
}
2、定时任务实现方式
引入quartz包
import org.quartz.Job;
import org.quartz.JobBuilder;
import org.quartz.JobDetail;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.quartz.SimpleScheduleBuilder;
import org.quartz.Trigger;
import org.quartz.TriggerBuilder;
import org.quartz.impl.StdSchedulerFactory; public class DeplayQuartzImpl implements Job{ @Override
public void execute(JobExecutionContext context) throws JobExecutionException {
//doing somethings
System.out.println("going scan database...");
} public static void main(String[] args) throws SchedulerException { //create task
JobDetail jobDetail = JobBuilder.newJob(DeplayQuartzImpl.class).withIdentity("job_1", "group_1").build(); //create trigger
Trigger trigger = TriggerBuilder.newTrigger()
.withIdentity("trigger_1", "group_trigger")
.withSchedule(SimpleScheduleBuilder.simpleSchedule()
.withIntervalInSeconds(3).repeatForever())
.build();
Scheduler scheduler = new StdSchedulerFactory().getScheduler(); //put task into trigger
scheduler.scheduleJob(jobDetail, trigger);
scheduler.start();
} }
实现java的Delayed接口简介版
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit; public class JdkDelayImpl implements Delayed{ private String orderId;
private long timeout; public JdkDelayImpl(String orderId, long timeout) {
this.orderId = orderId;
this.timeout = timeout + System.nanoTime();
} @Override
public int compareTo(Delayed delayed) {
if(delayed == this)
return 0;
JdkDelayImpl t = (JdkDelayImpl) delayed;
long d = (getDelay(TimeUnit.NANOSECONDS) - t.getDelay(TimeUnit.NANOSECONDS));
return (d == 0) ? 0 : ((d < 0) ? -1 : 1);
} @Override
public long getDelay(TimeUnit unit) {
return unit.convert(timeout - System.nanoTime(), TimeUnit.NANOSECONDS );
} void print() {
System.out.println(orderId + " order will being delete...");
} public static void main(String[] args) {
List<String> list = new ArrayList<String>();
list.add("001");
list.add("002");
list.add("003");
list.add("004");
list.add("005"); DelayQueue<JdkDelayImpl> queue = new DelayQueue<JdkDelayImpl>();
long start = System.currentTimeMillis();
for(int i = 0; i < 5; i++) {
queue.put(new JdkDelayImpl(list.get(i),
TimeUnit.NANOSECONDS.convert(3, TimeUnit.SECONDS)));
try {
queue.take().print();
System.out.println("after: " + (System.currentTimeMillis() - start)
+ " milliSeconds"); } catch (Exception e) {
}
}
} }
3、基于netty方式,添加netty包
import java.util.concurrent.TimeUnit; import io.netty.util.HashedWheelTimer;
import io.netty.util.Timeout;
import io.netty.util.Timer;
import io.netty.util.TimerTask; public class NettyDeplayImpl { static class MyTimeTask implements TimerTask { boolean flag; public MyTimeTask(boolean flag) {
this.flag = flag;
} @Override
public void run(Timeout arg0) throws Exception {
System.out.println("going to delete order...");
this.flag = false;
}
} public static void main(String[] args) {
MyTimeTask timeTask = new MyTimeTask(true);
Timer timer = new HashedWheelTimer();
timer.newTimeout(timeTask, 5, TimeUnit.SECONDS);
int i = 1;
while(timeTask.flag) {
try {
Thread.sleep(1000);
} catch (Exception e) {
}
System.err.println(i + " seconds gone");
i++;
}
}
}
java延迟队列的更多相关文章
- Java 延迟队列使用
延时队列,第一他是个队列,所以具有对列功能第二就是延时,这就是延时对列,功能也就是将任务放在该延时对列中,只有到了延时时刻才能从该延时对列中获取任务否则获取不到…… 应用场景比较多,比如延时1分钟发短 ...
- Spring Boot(十四)RabbitMQ延迟队列
一.前言 延迟队列的使用场景:1.未按时支付的订单,30分钟过期之后取消订单:2.给活跃度比较低的用户间隔N天之后推送消息,提高活跃度:3.过1分钟给新注册会员的用户,发送注册邮件等. 实现延迟队列的 ...
- 使用netty HashedWheelTimer构建简单延迟队列
背景 最近项目中有个业务,需要对用户新增任务到期后进行业务处理.使用定时任务定时扫描过期时间,浪费资源,且不实时.只能使用延时队列处理. DelayQueue 第一想到的是java自带的延时队列del ...
- rabbitmq延迟队列demo
1. demo详解 1.1 工程结构: 1.2 pom 定义jar包依赖的版本.版本很重要,rabbit依赖spring,两者必须相一致,否则报错: <properties> <sp ...
- JUC——延迟队列
所谓的延迟队列最大的特征是它可以自动通过队列进行脱离,例如:现在有一些对象被临时保存着,但是有可能该集合对象是一个公共对象,那么里面的某些数据如果不在使用的时候就希望其可以在指定的时间达到后自动的消失 ...
- Java阻塞队列(BlockingQueue)实现 生产者/消费者 示例
Java阻塞队列(BlockingQueue)实现 生产者/消费者 示例 本文由 TonySpark 翻译自 Javarevisited.转载请参见文章末尾的要求. Java.util.concurr ...
- RabbitMQ 延迟队列,消息延迟推送
目录 应用场景 消息延迟推送的实现 测试结果 应用场景 目前常见的应用软件都有消息的延迟推送的影子,应用也极为广泛,例如: 淘宝七天自动确认收货.在我们签收商品后,物流系统会在七天后延时发送一个消息给 ...
- Redis(二)延迟队列
1.目录 延迟队列 进一步优化 2.延迟队列 package com.redis; import java.lang.reflect.Type; import java.util.Set; impor ...
- Spring Boot (26) RabbitMQ延迟队列
延迟消息就是指当消息被发送以后,并不想让消费者立即拿到消息,而是等待指定时间后,消费者才拿到这个消息进行消费. 延迟队列 订单业务: 在电商/点餐中,都有下单后30分钟内没有付款,就自动取消订单. 短 ...
随机推荐
- [CodeForces - 1225E]Rock Is Push 【dp】【前缀和】
[CodeForces - 1225E]Rock Is Push [dp][前缀和] 标签:题解 codeforces题解 dp 前缀和 题目描述 Time limit 2000 ms Memory ...
- 特斯拉通过GitHub发布了这些源代码,用意何在?
特斯拉多年来一直在软件社区使用开源软件而不遵守许可证. 在迈向合规性的一步中,特斯拉现在正在发布其软件的一些部分,这对特斯拉黑客和安全研究人员来说非常有用. 特斯拉是一家软件强大的公司,它一直在使用大 ...
- grep正则表达式(一)
新建一批 txt 文件: [me@linuxbox ~]$ ls /bin > dirlist-bin.txt [me@linuxbox ~]$ ls /usr/bin > dirlist ...
- navicat12过期问题,Windows平台。
首先关闭Navicat 然后 win+R,输入regedit 回车,打开注册表编辑器: 删除HKEY_CURRENT_USER\Software\PremiumSoft\Data 展开HKEY_CUR ...
- android智能手机如何查看APK包名
工具/原料 智能手机一部 USB线一根 方法/步骤 1 首先.使用USB线,将电脑和手机连起来.注意.手机的USB调试默认需要打开,如下图所示. 2 然后启动电脑端的cmd应用,进入dos界面 ...
- 比传统事务快10倍?一张图读懂阿里云全局事务服务GTS
近日,阿里云全局事务服务GTS正式上线,为微服务架构中的分布式事务提供一站式解决方案.GTS的原理是将分布式事务与具体业务分离,在平台层面开发通用的事务中间件GTS,由事务中间件协调各服务的调用一致性 ...
- 在VMware中配置网卡之NAT模式
为什么要在VMware中配置网卡? 因为在远程连接服务器时,需要虚拟机连接网络 虚拟机网络配置的三种模式:桥接模式,NAT模式,主机模式 NAT模式也称之为网络转换模式,两层路由: 第一层路由:物理机 ...
- Python基础教程(021)--Pycharm简介
前言 学习Pycharm开发工具 内容 项目:就是一个功能复杂的软件 目标 必须掌握的工具
- Java总结第二期
大家好,我又来了!!啦啦,我知道你们很想我,很想我赶快写更多的文章来提高自己的水平,好吧,我就从了你们.下面跟我一起来光顾Java第二期,掌声,掌声!!! 第二章: 这章,我要给大家讲得内容有变量,常 ...
- 微信小程序相关操作
显示用户基本信息 在微信小程序中,经常会碰到需要展示微信用户的基本信息,如果只是为了显示用户信息,最简单有效的办法是使用open-data,这是微信小程序内置的用于展示微信开放数据的组件,通过改变ty ...