java使用DelayQueue實(shí)現(xiàn)延時(shí)任務(wù)
1、背景
項(xiàng)目中經(jīng)常會(huì)用到類似一些需要延遲執(zhí)行的功能,比如緩存。java提供了DelayQueue來很輕松的實(shí)現(xiàn)這種功能。Delayed接口中的getDelay方法返回值小于等于0的時(shí)候,表示時(shí)間到達(dá),可以從DelayQueue中通過take()方法取的到期的對(duì)象。到期對(duì)象是實(shí)現(xiàn)了Delayed的類。
2、demo
2.1 依賴配置
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!--非必須-->
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>5.8.0.M1</version>
</dependency>
<!--非必須-->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.22</version>
</dependency>
<!--非必須-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
2.2 整體架構(gòu)

** 工具類:**
執(zhí)行任務(wù)所需的基礎(chǔ)參數(shù)
import lombok.Data;
@Data
public class TaskBase {
//任務(wù)參數(shù),根據(jù)業(yè)務(wù)需求多少都行
private String identifier;
public TaskBase(String identifier) {
this.identifier = identifier;
}
}執(zhí)行的任務(wù)和時(shí)間
import cn.hutool.core.date.DateUtil;
import java.util.Date;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;
/**
* 延時(shí)任務(wù)
*/
public class DelayTask implements Delayed {
//任務(wù)參數(shù)
final private TaskBase data;
//任務(wù)的延時(shí)時(shí)間,單位毫秒
final private long expire;
/**
* 構(gòu)造延時(shí)任務(wù)
*
* @param data 業(yè)務(wù)數(shù)據(jù)
* @param expire 任務(wù)延時(shí)時(shí)間(ms)
*/
public DelayTask(TaskBase data, long expire) {
super();
this.data = data;
this.expire = expire + System.currentTimeMillis();
}
public TaskBase getData() {
return data;
}
public long getExpire() {
return expire;
}
@Override
public boolean equals(Object obj) {
if (obj instanceof DelayTask) {
return this.data.getIdentifier().equals(((DelayTask) obj).getData().getIdentifier());
}
return false;
}
@Override
public String toString() {
return "{" + "data:" + data.toString() + "," + "延時(shí)時(shí)間:" +expire+ DateUtil.format(new Date(),"yyyy.MM.dd HH:mm:ss") + "}";
}
@Override
public long getDelay(TimeUnit unit) {
return unit.convert(this.expire - System.currentTimeMillis(), unit);
}
@Override
public int compareTo(Delayed o) {
long delta = getDelay(TimeUnit.NANOSECONDS) - o.getDelay(TimeUnit.NANOSECONDS);
return (int) delta;
}
}
** 任務(wù)管理器:**
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;
import java.util.concurrent.DelayQueue;
import java.util.concurrent.Executors;
@Component
@Slf4j
public class DelayQueueManager implements CommandLineRunner {
private final DelayQueue<DelayTask> delayQueue = new DelayQueue<>();
private final Map<String, DelayTask> elements = new HashMap<>();
/**
* 加入到延時(shí)隊(duì)列中
*
* @param task
*/
public void put(DelayTask task) {
log.error("加入延時(shí)任務(wù):{}", task);
delayQueue.put(task);
}
/**
* 查詢延時(shí)任務(wù)
* @param taskID
* @return
*/
public DelayTask query(String taskID) {
return elements.get(taskID);
}
/**
* 取消延時(shí)任務(wù)
*
* @param task
* @return
*/
public boolean remove(DelayTask task) {
log.error("取消延時(shí)任務(wù):{}", task);
return delayQueue.remove(task);
}
/**
* 取消延時(shí)任務(wù)
*
* @param taskid
* @return
*/
public boolean remove(String taskid) {
return remove(new DelayTask(new TaskBase(taskid), 0));
}
@Override
public void run(String... args) throws Exception {
log.info("初始化延時(shí)隊(duì)列");
Executors.newSingleThreadExecutor().execute(new Thread(this::excuteThread));
}
/**
* 延時(shí)任務(wù)執(zhí)行線程
*/
private void excuteThread() {
while (true) {
try {
DelayTask task = delayQueue.take();
//執(zhí)行任務(wù)
processTask(task);
} catch (InterruptedException e) {
break;
}
}
}
/**
* 內(nèi)部執(zhí)行延時(shí)任務(wù)
*
* @param task
*/
private void processTask(DelayTask task) {
//獲取任務(wù)參數(shù),執(zhí)行業(yè)務(wù)task.getData().getIdentifier()
log.error("執(zhí)行延時(shí)任務(wù):{}-{}", task, task.getData().getIdentifier());
}
}
2.3 進(jìn)行測(cè)試
import com.example.demo.task.DelayQueueManager;
import com.example.demo.task.DelayTask;
import com.example.demo.task.TaskBase;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
@SpringBootTest
class DemoApplicationTests {
@Autowired
private DelayQueueManager delayQueueManager;
@Test
void contextLoads() throws InterruptedException {
//新增任務(wù)
delayQueueManager.put(new DelayTask(new TaskBase("abc"), 1000 * 1));
//新增任務(wù)
delayQueueManager.put(new DelayTask(new TaskBase("abc"), 1000 * 5));
//新增任務(wù)
delayQueueManager.put(new DelayTask(new TaskBase("abc"), 1000 * 6));
//測(cè)試任務(wù)需要下邊代碼執(zhí)行,線上不用
Thread.sleep(10 * 1000);
}
}

到此這篇關(guān)于java使用DelayQueue實(shí)現(xiàn)延時(shí)任務(wù)的文章就介紹到這了,更多相關(guān)DelayQueue延時(shí)任務(wù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot結(jié)合Redis實(shí)現(xiàn)序列化的方法詳解
Spring提供了一個(gè)RedisTemplate來進(jìn)行對(duì)Redis的操作,但是RedisTemplate默認(rèn)配置的是使用Java本機(jī)序列化。如果要對(duì)對(duì)象操作,就不是那么的方便。所以本文為大家介紹了另一種SpringBoot結(jié)合Redis實(shí)現(xiàn)序列化的方法,需要的可以參考一下2022-06-06
使用Java橋接模式打破繼承束縛優(yōu)雅實(shí)現(xiàn)多維度變化
這篇文章主要為大家介紹了使用Java橋接模式打破繼承束縛,優(yōu)雅實(shí)現(xiàn)多維度變化,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05
Java IO文件過濾器對(duì)命令設(shè)計(jì)模式的使用
java io流里面使用到了很多的設(shè)計(jì)模式,最典型的就是裝飾模式,還有命令模式,下面分兩部分來講Java IO文件過濾器對(duì)命令設(shè)計(jì)模式的使用,一起看看吧2017-06-06
Mybatis plus 配置多數(shù)據(jù)源的實(shí)現(xiàn)示例
這篇文章主要介紹了Mybatis plus 配置多數(shù)據(jù)源的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Java實(shí)現(xiàn)最小生成樹MST的兩種解法
最小生成樹(MST)指在連通圖的所有生成樹中,所有邊的權(quán)值和最小的生成樹。本文介紹了求最小生成樹的兩種方法:Prim算法和Kruskal算法,需要的可以參考一下2022-05-05

