Quartz---串行执行、JobDataMap持久化

1.Quartz串行执行

在Quartz中,作业(Job)默认是以并行方式执行的,这意味着如果调度器(Scheduler)有多个线程可用,并且满足触发条件,那么多个作业可能会同时执行。然而,有时候我们可能希望作业以串行方式执行,即一个接一个地执行,而不是同时执行。

实例

记录每个任务执行间隔

代码

package job;import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.quartz.JobKey;import javax.xml.crypto.Data;
import java.util.Date;public class MyJob implements Job {@Overridepublic void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {try {Thread.sleep(2000);System.out.println("excute:"+new Date());} catch (Exception e) {e.printStackTrace();}}
}
package schedule;import job.MyJob;
import org.quartz.*;
import org.quartz.impl.StdSchedulerFactory;public class MySchedule {public static void main(String[] args) {// 创建作业详情JobDetail jobDetail = JobBuilder.newJob(MyJob.class)//作业的类名.withIdentity("myJob", "group1")//作业的身份标识(名称和组名).build();//创建触发器Trigger trigger = TriggerBuilder.newTrigger().withIdentity("myTrigger", "group1")//触发器的身份标识(名称和组名).withSchedule(SimpleScheduleBuilder.simpleSchedule()//设置触发器的调度计划.withIntervalInSeconds(2)//设置作业执行的间隔时间为2秒。.repeatForever())//指定触发器应该无限次地重复执行作业。.build();try {// 创建调度器实例Scheduler scheduler = StdSchedulerFactory.getDefaultScheduler();// 将作业和触发器注册到调度器scheduler.scheduleJob(jobDetail, trigger);// 开始调度scheduler.start();} catch (SchedulerException se) {se.printStackTrace();}}
}

执行结果

每个任务最少需要3秒完成,而任务执行间隔为2,说明任务调度是并发执行的

串行执行的两种方法

1.使用单个线程调度器


如果你只有一个线程用于执行作业,那么作业自然会串行执行。你可以通过配置线程池来实现这一点,确保线程池的大小为1。这样,在任何给定时间,只有一个作业能够被执行。

ThreadPoolTaskExecutor threadPool = new ThreadPoolTaskExecutor(); threadPool.setCorePoolSize(1); threadPool.setMaxPoolSize(1); threadPool.setQueueCapacity(1); threadPool.setThreadNamePrefix("Quartz-Executor-"); threadPool.initialize(); SchedulerFactory schedulerFactory = new StdSchedulerFactory(); Scheduler scheduler = schedulerFactory.getScheduler(); scheduler.setThreadPool(threadPool); scheduler.start();

2.使用@DisallowConcurrentExecution注解
package job;import org.quartz.*;import javax.xml.crypto.Data;
import java.util.Date;
@DisallowConcurrentExecution
public class MyJob implements Job {@Overridepublic void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {try {Thread.sleep(3000);System.out.println("excute:"+new Date());} catch (Exception e) {e.printStackTrace();}}
}
3.结果

2.JobDataMap数据持久化(持久化到内存中)

实例

计算任务执行次数

代码

package job;import org.quartz.*;import java.util.Date;@DisallowConcurrentExecutionpublic class MyJob2 implements Job {@Overridepublic void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {try {JobDataMap jobDataMap = jobExecutionContext.getJobDetail().getJobDataMap();jobDataMap.put("count",jobDataMap.getInt("count")+1);System.out.println(jobDataMap.getInt("count")+"-"+new Date());} catch (Exception e) {e.printStackTrace();}}
}
package schedule;import job.MyJob;
import job.MyJob2;
import org.quartz.*;
import org.quartz.impl.StdSchedulerFactory;public class MySchedule2 {public static void main(String[] args) {Integer count=0;// 创建作业详情JobDetail jobDetail = JobBuilder.newJob(MyJob2.class)//作业的类名.withIdentity("myJob", "group1")//作业的身份标识(名称和组名).usingJobData("count", count).build();//创建触发器Trigger trigger = TriggerBuilder.newTrigger().withIdentity("myTrigger", "group1")//触发器的身份标识(名称和组名).withSchedule(SimpleScheduleBuilder.simpleSchedule()//设置触发器的调度计划.withIntervalInSeconds(1)//设置作业执行的间隔时间为2秒。.repeatForever())//指定触发器应该无限次地重复执行作业。.build();try {// 创建调度器实例Scheduler scheduler = StdSchedulerFactory.getDefaultScheduler();// 将作业和触发器注册到调度器scheduler.scheduleJob(jobDetail, trigger);// 开始调度scheduler.start();} catch (SchedulerException se) {se.printStackTrace();}}
}

执行结果

原因分析

Quartz的作业在执行时使用的是触发时创建的JobDataMap的一个拷贝,因此作业中对JobDataMap的任何更改都不会影响到原始的JobDataMap,也不会在作业执行结束后保留下来每次任务调度都会使用新的实例。

解决方案

任务类子类添加@PersistJobDataAfterExecution注解,并在任务类中实现Job接口

执行结果

注意此 注解对trigger中的JobDataMap无效

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/476082.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

沁恒CH32V30X学习笔记05--串口接收中断和空闲中断组合接收数据

同步异步收发器(USART)** 包含 3 个通用同步异步收发器(USART1/2/3)和 5 个通用异步收发器(UART4/5/6/7/8) 空闲帧,空闲帧是 10 位或 11 位高电平,包含停止位。 断开帧是 10 位或 11 位低电平,后跟着停止位 引脚模式配置 引脚分配 bsp 驱动代码 bsp_uart_it.c /…

【ArcGIS微课1000例】0103:导出点、线、面要素的折点坐标值

点要素对应的是一个或者若干个坐标,线要素对应的是对个坐标值对应的点连起来,面要素是多个坐标值对应的点连起来构成的封闭多边形。本文讲述导出点的坐标值。 文章目录 一、点要素坐标导出1. 计算点坐标2. 导出点坐标二、线要素坐标导出1. 生成线要素折点2. 计算折点坐标3. 导…

嵌入式第十七天!(文件IO)

文件IO: 标准IO和文件IO的区别: 1. 标准IO是库函数,是对系统调用的封装 2. 文件IO是系统调用,是Linux内核中的函数接口 3. 标准IO是有缓存的 4. 文件IO是没有缓存的 1. 操作步骤: 打开 -> 读/写 -> 关闭 2. 打开…

【Visual Studio】技巧 :自动与活动文档同步

在这里插入图片描述 工具 -> 选项 -> 项目和解决方案 - 勾选上面的 我厉害不!!!

【STM32 物联网】基础AT指令与基础Wifi功能AT指令

文章目录 前言一、基础AT指令1.1 测试AT启动1.2 重启模块1.3 查看版本信息1.4 进入深度睡眠模式1.5 开关回显1.6 恢复出场设置1.7 UART配置设置临时设置,不保存到Flash设置串口保存到Flash 1.8 设置sleep模式查询当前sleep模式设置当前sleep模式 二、基础Wifi功能AT…

OpenCV-42 直方图均匀化

目录 一、直方图均匀化原理 二、直方图均匀化在OpenCV中的运用 一、直方图均匀化原理 直方图均匀化是通过拉伸像素强度的分布范围,使得在0~255灰阶上的分布更加均匀,提高图像的对比度。达到改善图像主管视觉效果的目的。对比度较低的图像适合使用直方…

2024年 前端JavaScript入门到精通 第二天 笔记

2.1 赋值运算符 2.2 自增运算符 2.3 比较运算符 2.4 逻辑运算符以及优先级 2.5 JS基础Day2-23-if单分支语句以及判断成绩案例 2.6 JS基础Day2-24-if双分支语句以及判断润年案例 2.7 JS基础Day2-25-i侈分支语句以及上午总结 2.8 JS基础Day2-26-三元运算符以及求最大值案例 2.9 J…

在 MyBatis 中,可以使用相同的 SQL 映射语句进行批量删除和单个删除。

目录 前端代码: 后端代码: controller service层接口 service接口的实现 mapper层接口 xml sql 效果:(点击操作列的删除,可删除一行数据。勾选多个多选框再点击批量删除,可删除多个) …

PyCharm 主题和字体 (Scheme Editor Font)

PyCharm 主题和字体 [Scheme & Editor Font] References Scheme & Editor Font File -> Settings -> Editor -> Colors & Fonts -> Font Show only monospaced fonts: 只显示等宽字体。编程时使用等宽字体效果较好。 References [1] Yon…

Opencv实战(1)读取与图像操作

Opencv 文章目录 Opencv一、读取图片1.imshow2.namedWindow3.imshow4.效果图 二、像素操作(1).访问像素1. at()2.Mat_ (2).遍历像素1.指针遍历2.迭代器遍历 (3).threshold(4).通道分离1.split2.merge (5)Gamma矫正 三、深浅拷贝 一、读取图片 1.imshow Mat imread(const stri…

Spark---环境搭建---入门概念

目录 环境搭建 测试 Apache Spark是用于大规模数据处理的统一分析引擎; spark 仅仅替代了hadoop的mapraduce; spark比hadoop快一百倍; 环境搭建 1:解压; 2:配置spark环境变量: vim /etc/pro…

查看 PyCharm 代码文件目录位置

查看 PyCharm 代码文件目录位置 1. Show in Files2. Copy PathReferences 1. Show in Files right click -> Show in Files / Show in Explorer 即可打开目录 2. Copy Path right click -> Copy Path 即可复制目录或文件路径 References [1] Yongqiang Cheng, http…