SpringBoot 异步延时任务

1. 引言

异步延时任务在日常工作中是比较常见的。实现方案也比较多。

2. 延迟队列DelayQueue

延迟队列是 jdk1.5 提供的,核心是优先级队列PriorityQueue

2.1 延时任务

DelayQueue要求队列中的元素必须实现Delayed接口。

import java.util.Date;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;

public class DelayTask<T> implements Delayed {
    final private T data;
    final private long expire;

    /**
     * 构造延时任务
     * @param data      业务数据
     * @param expire    任务延时时间(ms)
     */
    public DelayTask(T data, long expire) {
        super();
        this.data = data;
        this.expire = expire + System.currentTimeMillis();
    }

    public T getData() {
        return data;
    }

    public long getExpire() {
        return expire;
    }


    @Override
    public String toString() {
        return "{" + "data:" + data.toString() + "," + "expire:" + new Date(expire) + "}";
    }

    @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;
    }
}

2.2 延时任务管理器

实现CommandLineRunner接口,重写了方法void run(String... args);,这个方法会在Spring容器启动后,所有Bean都已经被创建和初始化之后被调用。这个方法的参数是一个字符串数组,通常包含命令行参数,但是你可以根据需要忽略这个参数。
定义一个静态变量DelayQueue,用于保存延时任务。
在 run 方法中启用线程循环从队列中获取数据,获取不到就阻塞线程,直到有数据为止。
取到任务后,执行processTask方法出来业务逻辑。

@Component
@Slf4j
public class DelayQueueManager implements CommandLineRunner {

    private static DelayQueue<DelayTask> delayQueue = new DelayQueue<>();

    /**
     * 加入到延时队列中
     *
     * @param task
     */
    public void put(DelayTask task) {
        log.info("加入延时任务:{}", task);
        delayQueue.put(task);
    }

    /**
     * 取消延时任务
     *
     * @param task
     * @return
     */
    public boolean remove(DelayTask task) {
        log.info("取消延时任务:{}", task);
        return delayQueue.remove(task);
    }


    @Override
    public void run(String... args) throws Exception {
        log.info("初始化延时队列");
        Executors.newSingleThreadExecutor().execute(new Thread(this::excuteThread));
    }

    /**
     * 延时任务执行线程
     */
    private void excuteThread() {
        while (true) {
            try {
                DelayTask task = delayQueue.take();
                processTask(task);
            } catch (InterruptedException e) {
                break;
            }
        }
    }

    /**
     * 内部执行延时任务
     *
     * @param task
     */
    private void processTask(DelayTask task) {
        log.info("执行延时任务:{}", task);
        //根据task中的data自定义数据来处理相关逻辑,例 if (task.getData() instanceof XXX) {}
        // do something
    }
}

相关推荐

  1. SpringBoot 异步任务

    2024-04-14 12:38:01       38 阅读
  2. 基于Redis实现任务

    2024-04-14 12:38:01       19 阅读
  3. Springboot 集成Rabbitmq之队列

    2024-04-14 12:38:01       15 阅读
  4. SpringBoot异步任务获取HttpServletRequest

    2024-04-14 12:38:01       24 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-04-14 12:38:01       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-04-14 12:38:01       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-04-14 12:38:01       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-04-14 12:38:01       20 阅读

热门阅读

  1. Asp.net 使用了 bootstrap,发布时样式丢失了

    2024-04-14 12:38:01       20 阅读
  2. Hadoop MapReduce解析

    2024-04-14 12:38:01       17 阅读
  3. ChatGPT指导下的学术写作:打造高质量论文

    2024-04-14 12:38:01       19 阅读
  4. Swift中的类

    2024-04-14 12:38:01       23 阅读
  5. python 有什么资格称为编程排行榜第一

    2024-04-14 12:38:01       54 阅读
  6. 6.0 MapReduce 使用

    2024-04-14 12:38:01       15 阅读