背景
最近项目中有个业务,需要对用户新增任务到期后进行业务处理。使用定时任务定时扫描过期时间,浪费资源,且不实时。只能使用延时队列处理。
DelayQueue
第一想到的是java自带的延时队列delayqueue。
首先实现一个Delyed类。
实现两个最重要方法。第一个是队列里面的消息排序。DelayQueue底层使用的是阻塞队列。队列的消费端会去take队列的头部元素,没有元素就阻塞在那里。因此,延迟队列中的元素必须按执行时间顺序排列。
@Override public int compareTo(Delayed delayed) { Message message = (Message) delayed; return this.exceptTime > message.getExceptTime() ? 1 : 0; }
第二个方法是剩余时间延迟时间。每加入一个元素时将延迟时间传入,得到一个预期执行时间。每当执行此方法的时候,使用预期时间减去当前时间,即时剩余延迟时间。换句话说,还有多长时间执行。为0时立即执行。
@Override public long getDelay(TimeUnit unit) { System.out.println(exceptTime - System.nanoTime()); return unit.convert(exceptTime - System.nanoTime(), TimeUnit.SECONDS); }
全部代码:
import java.util.concurrent.Delayed; import java.util.concurrent.TimeUnit; public class Message implements Delayed{ private Integer id; private String content; private long delay;//延迟时间 private long exceptTime;//执行时间 public Message() {} public Message(Integer id, String content, long delay) { this.id = id; this.content = content; this.delay = delay; this.exceptTime = System.nanoTime() + delay; } @Override public int compareTo(Delayed delayed) { Message message = (Message) delayed; return this.exceptTime > message.getExceptTime() ? 1 : 0; } @Override public long getDelay(TimeUnit unit) { System.out.println(exceptTime - System.nanoTime()); return unit.convert(exceptTime - System.nanoTime(), TimeUnit.SECONDS); } public String getContent() { return content; } public void setContent(String content) { this.content = content; } public Integer getId() { return id; } public void setId(Integer id) { this.id = id; } public long getDelay() { return delay; } public void setDelay(long delay) { this.delay = delay; } public long getExceptTime() { return exceptTime; } public void setExceptTime(long exceptTime) { this.exceptTime = exceptTime; } }