当前位置: 首页 > news >正文

【小工具】定时任务执行器

定时任务执行器

  • 背景
  • 版本
  • 代码
    • Job
    • Job执行机

背景

有时我们的项目内需要一个定时执行器来执行某些任务,就需要一个简单好用的定时任务机。
注意,这个定时任务机并不原生支持分布式,如果需要分布式的功能请自己实现。

版本

  • jdk21

代码

Job

用于统一封装需要执行的任务和开始时间、间隔时间

import lombok.Getter;
import java.util.Objects;

/**
 * 任务封装
 */
@Getter
public class Job implements Comparable<Job> {

    /**
     * 待执行任务
     */
    private Runnable task;
    /**
     * 下次开始时间
     */
    private long startTime;
    /**
     * 需要等待时间
     */
    private long delay;


    /**
     * 私有无参构造器,防止外部调用
     */
    private Job() {
    }

    /**
     * 从startTime开始,每隔delay毫秒执行一次task
     *
     * @param task      待执行任务
     * @param startTime 开始时间
     * @param delay     等待时间
     */
    public Job(Runnable task, long startTime, long delay) {
        if (Objects.isNull(task)) {
            throw new IllegalArgumentException("待执行任务不能为Null");
        }
        if (startTime <= 0) {
            throw new IllegalArgumentException("开始时间非法");
        }
        if (delay <= 0) {
            throw new IllegalArgumentException("等待时间非法");
        }
        this.task = task;
        this.startTime = startTime;
        this.delay = delay;
    }

    /**
     * 用于排序任务
     *
     * @param o the object to be compared.
     * @return 排序结果
     */
    @Override
    public int compareTo(Job o) {
        return Long.compare(this.startTime, o.startTime);
    }
}

Job执行机

import com.utils.ScheduleUtil.Job;
import org.slf4j.Logger;

import java.util.concurrent.*;
import java.util.concurrent.locks.LockSupport;

public class MineSchedule {

    // 注意,这里的线程池默认只给了6个空间,是为了方便学习。实际生产中应当做更精确的线程池,比如用google提供的线程池创建工具
    private final ExecutorService service = Executors.newFixedThreadPool(6);
    private final Trigger trigger = new Trigger();

    class Trigger {

        private static final Logger log = org.slf4j.LoggerFactory.getLogger(Trigger.class);

        /**
         * 优先级队列,会自动排序
         */
        PriorityBlockingQueue<Job> queue = new PriorityBlockingQueue<>();

        Thread machine = new Thread(() -> {
            while (true) {
                // 如果队列中没有任务,就park
                while (queue.isEmpty()) {
                    log.info("队列中没有任务,线程park");
                    LockSupport.park();
                }
                // 如果队列中有任务,就取出最早的任务,判断是否到时间了,如果到时间了,就执行任务,否则就park
                // peek和poll的区别是,peek不会删除元素,poll会删除元素
                // 所以用peek先把队列的头部取出来看一眼时间做if判断
                Job latelyJob = queue.peek();
                if (latelyJob.getStartTime() < System.currentTimeMillis()) {
                    // 需要执行时才poll出来执行
                    latelyJob = queue.poll();
                    if (latelyJob != null) {
                        service.execute(latelyJob.getTask());
                        queue.offer(rebuildJob(latelyJob));
                    }
                } else {
                    LockSupport.parkUntil(latelyJob.getStartTime());
                }
            }
        }, "scheduler-machine");

        {
            machine.start();
            log.info("触发器启动");
        }

        // 添加任务立即执行一次,所以需要一个强制唤醒
        void wakeUp() {
            LockSupport.unpark(machine);
        }

        // 任务重新放回队列,等候下一次执行
        private Job rebuildJob(Job old) {
            return new Job(old.getTask(), old.getStartTime() + old.getDelay(), old.getDelay());
        }

    }

    /**
     * 每隔delay毫秒数,自动执行一次task
     *
     * @param task  需要周期执行的任务
     * @param delay 延迟时间
     */
    public void schedule(Runnable task, long delay) {
        // 最开始的想法,搞一个线程池,每次有新任务的时候把任务丢进去,睡delay毫秒后执行
        // 但是这是有问题的,线程耗尽就完了,而且线程不可复用,创建线程消耗资源很大
        // 那我们就考虑这么一种设计:
        // 1. 有一个定时触发器,每隔delay时间被唤醒,然后去尝试执行任务
        // 2. 线程池只负责执行任务,不负责处理时间
        // 那么这个触发器需要什么信息呢?第一,所有需要执行的任务,第二,需要delay的时间
        // 那么我们封装一个Job类,专门用来记录任务和时间
        // 再写一个trigger,用于时间触发
        Job job = new Job(task, System.currentTimeMillis(), delay);
        trigger.queue.offer(job);
        trigger.wakeUp();
    }
}

相关文章:

  • CUDA编程高阶优化:如何突破GPU内存带宽瓶颈的6种实战策略
  • 深入详解MYSQL的MVCC机制
  • 第一章 教育与教育学
  • 提权实战!
  • Python 基础语法汇总
  • 数据运营与数据分类
  • CPU(中央处理器)
  • 28.[MRCTF2020]Xor1(保姆教程)
  • 基于PySide6与pycatia的CATIA绘图比例智能调节工具开发全解析
  • [Dify] Dify 本地部署及连接 Ollama 模型全流程指南
  • Deepseek IP-Adapter与InstantID的区别
  • 计算机系统概论
  • CS5346 - Visualization Design Process
  • 泛型的二三事
  • oracle DECODE 函数
  • DeeplxFile相关文件下载
  • 七、自动化概念篇
  • Elasticsearch 故障转移及水平扩容
  • 使用Python和Matplotlib可视化字体轮廓:从路径数据到矢量图形
  • C语言中三角与反三角函数的表达
  • 海昏侯博物馆展览上新,“西汉帝陵文化展”将持续展出3个月
  • 广西壮族自治区党委常委会:坚决拥护党中央对蓝天立进行审查调查的决定
  • 俄媒:俄乌代表团抵达谈判会场
  • 英国6月初将公布对华关系的审计报告,外交部:望英方树立正确政策导向
  • 王东杰评《国家与学术》︱不“国”不“故”的“国学”
  • 四川甘孜炉霍县觉日寺管委会主任呷玛降泽被查