<Java 定时任务解决方案Quartz>学习笔记

2023-10-29

理论基础

小顶堆

在这里插入图片描述

参考文章:
数据结构&算法-----(2)高级数据结构
的部分

在这里插入图片描述

定时任务中每一个任务Job都对应堆中的一个节点, 对应定时任务的到期时间
Delay 延迟时间 到期时间
每次取堆顶元素执行

注意:
在添加/删除元素时候需要保证根节点 最大/最小

① 向堆中添加元素
在这里插入图片描述
② 取出堆中的最大 / 最小 (堆顶) 元素

在这里插入图片描述

时间轮算法

普通时间轮

使用链表or数组实现

思考: 删除堆顶元素中的下沉操作, 在树的节点很多的时候会比较耗性能. 而且在定时任务中取任务是一个很频繁的操作

下沉其实是为了取出小顶堆中最小元素, 每次下沉进行元素比对会造成性能浪费


参考文章:
那些惊艳的算法们(三)—— 时间轮

while-true-sleep

在这里插入图片描述

核心思想: 我只需要把任务放到它需要被执行的时刻,然后等着时针转到这个时刻时,取出该时刻放置的任务,执行就可以了

遍历数组, 每个下标放置一个链表, 链表节点放置任务, 遍历到了就取出执行

多一个维度: 分 / 秒 / 天…, 刻度会成倍的增加

Round型时间轮

任务上记录一个round, 遍历到了就将 round-1, 为0时取出执行就可以了
但是需要遍历所有的任务, 效率较低

分层时间轮

cron表达式的底层原理

使用多个不同时间维度的轮, 天轮: 记录几点执行; 月轮, 记录几号执行
月轮遍历到了, 将任务取出放到天轮里边, 即可以实现几号几点执行

分层时间轮: 避免遍历所有元素

JDK定时器: Timer使用及原理分析

public class Timer {

	// 用于存放任务, 小顶堆, getMin方法取出任务
	private final TaskQueue queue = new TaskQueue();
	
	// 用于执行任务的线程
	private final TimerThread thread = new TimerThread(queue);
	
	// 调用start()方法, 触发多线程执行. 以多线程形式运行 TimerThread 中的 run 方法
	// 注意: 并不是多线程运行业务逻辑, 业务逻辑在TimerTask里面
    public Timer(String name) {
        thread.setName(name);
        thread.start();
    }

	// schedule方法会触发调用
    private void sched(TimerTask task, long time, long period) {
        if (time < 0)
            throw new IllegalArgumentException("Illegal execution time.");
            
        if (Math.abs(period) > (Long.MAX_VALUE >> 1))
            period >>= 1;

        synchronized(queue) { // 锁了一下任务队列
            if (!thread.newTasksMayBeScheduled)
                throw new IllegalStateException("Timer already cancelled.");

            synchronized(task.lock) { // Task可以往多个Timer加, 使用双重校验
                if (task.state != TimerTask.VIRGIN)
                    throw new IllegalStateException(
                        "Task already scheduled or cancelled");
                task.nextExecutionTime = time;
                task.period = period;
                task.state = TimerTask.SCHEDULED;
            }

            queue.add(task); // 将task放入queue, 调用notify唤醒队列
            if (queue.getMin() == task)
                queue.notify();
        }
    }

	// 有多个重载, 可以指定确定的时间也可以指定相对的时间
    public void schedule(TimerTask task, Date firstTime, long period) {
        if (period <= 0)
            throw new IllegalArgumentException("Non-positive period.");
        sched(task, firstTime.getTime(), -period);
    }
	
	// schedule和scheduleAtFixedRate区别:
	// schedule 预设的执行时间 nextExecutTime 12:00:00, 12:00:02, 12:00:04
    // schedule 真正的执行时间 取决上一个任务的结束时间 ExecutTime   03  05  08  丢任务(少执行了次数)
    // scheduleAtFixedRate  严格按照预设时间 12:00:00  12:00:02  12:00:04(执行时间会乱)
    public void scheduleAtFixedRate(TimerTask task, Date firstTime, long period) {
        if (period <= 0)
            throw new IllegalArgumentException("Non-positive period.");
        sched(task, firstTime.getTime(), period);
    }
}

class TimerThread extends Thread {
    public void run() {
        try {
            mainLoop();
        } finally {
            synchronized(queue) {
                newTasksMayBeScheduled = false;
                queue.clear(); // 任务执行完后清理队列
            }
        }
    }

    private void mainLoop() {
        while (true) {
            try {
                TimerTask task; // 接收一个TimerTask
                boolean taskFired;
                synchronized(queue) {
                    while (queue.isEmpty() && newTasksMayBeScheduled) 
                        queue.wait(); // 队列为空时阻塞, 放一个任务进来会notify解除阻塞
                    if (queue.isEmpty())
                        break;
                    long currentTime, executionTime;
                    task = queue.getMin(); // 从小顶堆中取出Task
                    synchronized(task.lock) {
                        if (task.state == TimerTask.CANCELLED) {
                            queue.removeMin(); // 从堆顶取出来后就会删掉任务
                            continue;
                        }
                        currentTime = System.currentTimeMillis(); // 当前时间
                        // 获取Task的下一次执行时间, 重新加入堆. 下一次时间点的任务相当于是一个新的任务
                        executionTime = task.nextExecutionTime; 
                        if (taskFired = (executionTime<=currentTime)) {
                            if (task.period == 0) {
                                queue.removeMin();
                                task.state = TimerTask.EXECUTED;
                            } else {
                                queue.rescheduleMin( // 将新的任务重新入队
                                // 如果后面task.run();方法超时, 有可能会导致任务的丢失
                                task.period<0 ? currentTime   - task.period : executionTime + task.period); 
                            }
                        }
                    }
                    if (!taskFired)
                        queue.wait(executionTime - currentTime);
                }
                if (taskFired)
                    task.run(); // task直接调的run方法, 并不是通过start启动的
                    // 相当于是以单线程的方式执行, 因为任务超时会导致任务阻塞
                    // 解决方案: 在TimerTask的run()方法中使用线程池去执行
            } catch(InterruptedException e) {
            }
        }
    }
}

严重依赖系统时间
不能给定相对时间执行, eg: 每周一执行找不到
在这里插入图片描述

定时任务线程池解析

在这里插入图片描述

Leader - Follower 模式 (定时任务线程池中的设计模式)

在这里插入图片描述

定时任务框架 - Quartz

在这里插入图片描述

注意, 同一个job定义, 执行的是多个实例
通过System.identityHashCode打印出hashcode值不一样
每一次任务的jobDataMap也不同

在这里插入图片描述
取的map不同, 每次打印都是1

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

<Java 定时任务解决方案Quartz>学习笔记 的相关文章

  • ThreadPoolTaskScheduler实现动态管理定时任务

    最近 有个项目有需要用到定时任务 所以做了一个动态管理定时任务的模块 本文将从项目背景 需求 选型 思路 具体实现等方面展开介绍 背景 有个支付类的项目 中间会产生一些中间态的订单 需要有个定时任务轮询确认订单状态 该类项目体量较小 单节点
  • 宝塔定时计划任务

    选择shell脚本 选择执行周期 在脚本内容内写入 curl sS connect timeout 10 m 60 此处为地址链接
  • ScheduledThreadPoolExecutor有坑嗷~

    概述 最近在做一些优化的时候用到了ScheduledThreadPoolExecutor 虽然知道这个玩意 但是也很久没用 本着再了解了解的心态 到网上搜索了一下 结果就发现网上有些博客在说ScheduledThreadPoolExecut
  • SpringBoot第47讲:SpringBoot定时任务 - Netty HashedWheelTimer方式

    SpringBoot第47讲 SpringBoot定时任务 Netty HashedWheelTimer方式 timer 和 ScheduledExecutorService 是JDK内置的定时任务方案 而业内还有一个经典的定时任务的设计叫
  • linux的crontab定时配置全过程

    今天因为业务需求 需要在服务器上配置定时任务执行数据库的sql语句 所以使用到了linux 的crontab 特此记录 分享一下自己的过程 首先呢 我们知道crontab是在linux中用于设定在某些时间可以自动执行某些命令的功能 在lin
  • 定时任务Schedule的使用

    定时任务或者说定时调度 是系统中比较普遍的一个功能 例如数据归档 清理 数据定时同步 非实时 定时收发 流量控制等等都需要用到定时任务 常见的定时调度框架有Quartz TBSchedule等 同样 Spring自3 0版本起也增加了任务调
  • Springboot 整合Quartz

    目录 一 Quartz介绍 二 Quartz核心元素 1 Scheduler 任务调度器 2 Trigger 触发器 3 JobDetail 定时任务的信息载体 三 Springboot整合Quartz 1 添加Quartz依赖 2 app
  • java corn 定时任务调度,每分钟执行一次,每半个小时执行一次

    java corn 表达式 每分钟执行一次 Scheduled cron 0 1 每半个小时执行一次 Scheduled cron 0 0 30 springboot 类 EnableScheduling Configuration Slf
  • SpringBoot:@Schedule定时任务

    一 Schedule SpringBoot内置了Sping Schedule定时框架 通过注解驱动方式添加所注解方法到定时任务 根据配置定时信息定时执行 二 定时任务实现 1 开启定时任务 package com gupao springb
  • 实现一个任务调度系统,这篇文章就够了

    阅读一篇 定时任务框架选型 的文章时 一位网友的留言电到了我 我看过那么多所谓的教程 大部分都是教 如何使用工具 的 没有多少是教 如何制作工具 的 能教 如何仿制工具 的都已经是凤毛麟角 中国 软件行业 缺的是真正可以 制作工具 的程序员
  • Java定时任务技术分析

    从零打造项目 系列文章 工具 比MyBatis Generator更强大的代码生成器 ORM框架选型 SpringBoot项目基础设施搭建 SpringBoot集成Mybatis项目实操 SpringBoot集成MybatisPlus项目实
  • Flutter中的Timer

    Flutter中的Timer 1 单次运行的定时器 2 重复运行定时器 Timer的官方文档资料 1 单次运行的定时器 源码 factory Timer Duration duration void Function callback if
  • python笔记15--常见定时功能

    python笔记15 常见定时功能 1 简介 2 定时方法 最粗暴的 while sleep schedule 实现定时 APScheduler 后台定时 3 注意事项 4 说明 1 简介 实际项目中经常要执行一些定时任务 因此有必要按需选
  • 使用Java定时执行shell脚本

    执行shell脚本 Runtime getRuntime exec 可以直接执行部分命令 不过执行一个shell脚本的话更方便修改 public static void runshell String path try String get
  • java中的实时定时任务管理系统

    java中的实时定时任务管理系统 前言 新增定时任务的线程池配置类 配置ScheduledFuture的包装类 ScheduledFuture是ScheduledExecutorService定时任务线程池的执行结果 实现Runnable接
  • flask+APScheduler定时任务的使用

    APScheduler定时任务使用以及在flask中的调用 APScheduler简介 组成部分 调度器 安装 普通使用安装 结合flask使用安装 使用 添加job add job参数详解 interval 间隔时间 每隔一段时间执行 d
  • Linux实现使用定时任务执行php程序(以及定时任务url带参数)

    php程序已经写好了 位置 data html XXX redis to mysql php php安装位置为 app bin php 查找php安装位置使用 whereis php which php php v which 这条命令主要
  • centos7 linux定时任务详解

    前言 工作中需要开启一个定时任务 每天晚上2点进行爬虫代码的运行 这不得不去学习一下linux 下的定时任务crontab crontab yum install crontabs 说明 sbin service crond start 启
  • [114]python supervisor使用

    Supervisor 是基于 Python 的进程管理工具 只能运行在 Unix Like 的系统上 也就是无法运行在 Windows 上 Supervisor 官方版目前只能运行在 Python 2 4 以上版本 但是还无法运行在 Pyt
  • Windows系统设置每天自动备份指定文件并自动删除七天前的文件(脚本+Windows任务计划)

    Windows系统设置每天自动备份指定文件并自动删除七天前的文件 脚本 Windows任务计划 在生活中和工作中有时候为了避免电脑宕机导致文件丢失常常会使用一些方式去备份文件 今天小编给大家介绍一个方法 用于自动备份指定文件 并删除N天前的

随机推荐