分布式定时任务

article/2025/10/10 4:43:55

分布式定时任务

1,什么是分布式定时任务;
2,为什么要采用分布式定时任务
3,怎么样设计实现一个分布式定时任务
4,当前比较流行的分布式定时任务框架

1,什么是分布式定时任务

  • 首先,我们要了解计划任务这个概念,计划任务是指由计划的定时运行或者周期性运行的程序。我们最常见的就是Linux的‘crontab’和Windows的‘计划任务’。

  • 那么什么是分布式定时任务,个人总结为:把分散的,可靠性差的计划任务纳入统一的平台,并实现集群管理调度和分布式部署的一种定时任务的管理方式。叫做分布式定时任务。

2,为什么要采用分布式定时任务

单点定时任务的缺点:

  • 功能相对简单,交互性差,任务部署效率低,开发和维护成本比较高,不能很好的满足各系统定时任务的管理和控制,尤其在多系统的环境下更加明显;
  • 许多任务都是单机部署,可用性差;
  • 任务跟踪和告警难以实现。

分布式定时任务的优势

  • 通过集群的方式进行管理调度,大大降低了开发和维护成本;
  • 分布式部署,保证了系统的高可用性,伸缩性,负载均衡,提高了容错;
  • 可以通过控制台部署和管理定时任务,方便灵活高效;
  • 任务都可以持久化到数据库,避免了宕机和数据丢失带来的隐患,同时有完善的任务失败重做机制和详细的任务跟踪及告警策略。

3,怎么样设计和实现一个分布式定时任务
3.1 分时方案

  • 严格划分时间片,交替运行计划任务,当主系统宕机后,备用系统仍然工作,但是处理初期被拉长了。
  • 缺点:周期延长了。

untitled.png

3.2 HA高可用方案:

  • 正常情况下主系统工作,备用系统守候,心跳检测发现主系统出现故障备用系统启动。
  • 缺点:单一系统,不能做负载均衡,只能垂直扩展,也就是硬件层面的升级,无法做水平扩展。

untitled1.png

3.3 多路心跳方案:

  • 采用多路心跳,做服务级,进程级的,IP和端口级别的心跳检测,正常情况是主系统工作,备用系统守候,心跳检测主系统出现故障,备用系统启动,当再次检测到主系统工作,则将执行权交回主系统。
  • 缺点:开发比较复杂,程序健壮性要求高。

untitled2.png

3.4 任务抢占方案:

  • A,B两台服务器同时工作,启动需要存在一前一后,谁先启动谁率先加锁,其他服务器只能等待,他们同时对互斥锁进行监控,一旦发现锁被释放,其他服务那个先抢到,那个运行,运行前加排他锁。
  • 优点:可以进一步实现多服务器横向扩展。
  • 缺点:开发复杂,程序健壮性要求高,有时候会出现不释放锁的问题。

untitled4.png

3.5 任务轮询或任务轮询+抢占排队方案

  • 每个服务器首次启动时加入队列;
  • 每次任务运行首先判断自己是否是当前可运行任务,如果是便运行;
  • 如果不是当前运行的任务,检查自己是否在队列中,如果在,便推出,如果不在队列中,便键入队列。

untitled5.png

通过以上这些方案,可以看出3.5的方案才是优先选择的,扩展性好,开发复杂度不是很高。那么这种方案需要的需要的技术原理是什么呢,那就是分布式互斥锁和队列。

3.6 原理

  • 分布式互斥锁:
    互斥锁也叫排他锁,用于并发时管理多进程和多进程同一时刻只能有一个进程或者线程操作一个功能。我们将进程,线程中的锁延伸到互联网上,实现对一个节点运行的进程或线程加 锁,解锁操作。这样便能控制节点上的进程或线程的并发。如下图:

untitled7.png

有两台服务器运行定时任务,其中serverA的T2做了加锁操作,其他程序必须等它释放锁了才能运行。 那么如果serverA在加锁的过程中,出现宕机怎么办,是否会一直处于别锁状态。那么我们可以在每个锁都设置一个超时阈值,一旦超时便自动解锁。这样就不会因为宕机导致锁一直不被释 放。另外我们还要考虑命名空间的问题,主要是防止出现同名锁,导致被覆盖。

  • 队列:
    在上面的基础上,排队运行任务。

untitled8.png

从上图中可以看出,TaskQueue中排队情况,运行是自上而下的,当然这个顺序可以自己设置规则,只需要先进先出的远程即可。另外,Task Queue我们需要做至少两个节点,他们遵循主 从结构的原则,主节点需要实时向从节点同步数据,保证在主节点不可用,从节点可以替代。当然,这里可以使用权重轮询的方式,加上数据异步同步,让所有节点都可以做主从的切换, 根据运行状况来分配,可能会更好,但是这样开发难度也有所提高,但是大大增加了高可用性。

3.7 总结:

  • 最后,我们要根据我们实际的情况,需要提供数据库和缓存方面的一些配套服务,这里就不做详解;

  • 这样我们整体的一个分布式定时任务平台就可以实现了,就可以保证计划任务的分布式运行。

4,当前比较流行的分布式定时任务框架:
4.1 Quartz:

  • Quartz是Java领域最著名的开源任务调度工具。Quartz提供了极为广泛的特性如持久化任务,集群和分布式任务
  • 特点:
    • 完全由Java写成,同时可以很方便的和java的另外一个框架spring集成;
    • 强大的调度功能:支持丰富多样的调度方法,可以满足各种常规及特殊需求;
    • 灵活的应用方式:支持任务和调度的多种组合方式,支持调度数据的多种存储方式;
    • 分布式和集群能力,负载均衡和高可用性;

4.2 Elastic-job:

  • Elastic-Job是ddframe中dd-job的作业模块中分离出来的分布式弹性作业框架。去掉了和dd-job中的监控和ddframe接入规范部分。该项目基于成熟的开源产品Quartz和 Zookeeper及其客户端Curator进行二次开发

  • 项目开源地址:https://github.com/dangdangdotcom/elastic-job

  • 特点:

    • 定时任务:基于成熟的定时任务作业框架Quartz cron表达式执行定时任务;
    • 作业注册中心:基于Zookeeper和其客户端Curator实现全局作业注册控制中心。用于注册,控制和协调分布式作业执行。
    • 作业分片:将要给任务分片成多个小任务项到多服务器上同时执行;
    • 弹性扩容缩容:运行中的作业服务器崩溃,或新增N台作业服务器,作业框架将在下次作业执行前重新分片,不影响当前作业执行;
  • 支持多种作业执行模式:支持OneOff,Perpetual和SequenecePerpetual三种作业模式;

  • 失效转移:运行中的作业服务器崩溃不会导致重新分片,只会在下次作业启动时分片。启用失效转移功能可以在本次作业执行过程中,监测其他作业服务器空闲,抓取未完成的孤儿分片项 执行;

  • 运行时状态收集:监控作业运行时状态,统计最近一段时间处理的数据成功和失败数量,记录作业上次运行开始时间,结束时间和下次运行时间;

  • 作业停止,恢复和禁用:用于操作作业启动和停止,并可以禁止某作业运行,一般在上线时常用;

  • 被错过执行的作业重触发:自动记录错过执行的作业,并在上次作业完成后自动触发。

  • 多线程快速处理数据:使用多线程处理抓取到的数据,提升吞吐量;

  • 幂等性:重复作业任务项判定,不重复执行已运行的作业任务项;

  • 容错处理:作业服务器和Zookeeper服务器通信失败后则立即停止作业运行,防止作业注册中心将失效的分片分项配给其他作业服务器,而当前作业服务器任在执行任务,导致重复执行。

  • Spring支持:支持Spring容器,自定义命名空间,支持占位符;

  • 运维平台:提供了运维平台,可以管理作业和注册中心。

从以上可以看出Elastic-job是在Quartz的基础上又做了一次全面的升级,做了配套的周边基础服务工作,完全成为了一个成熟的分布式定时任务框架。后面会分别介绍Quartz和 Elastic-job的详细原理和具体的使用方法。



///

 

Quartz应用和集群原理分析

使用的环境版本:spring4.x+quartz2.2.x

****1.1 如何在spring中集成quartz集群****

1.1.1 基于maven项目,需要在pom.xml引入的j依赖为:

<dependency><groupId>org.quartz-scheduler</groupId><artifactId>quartz</artifactId>
</dependency>

1.1.2 Quartz集群的基本配置信息:命名为quartz.properties

#调度标识名 集群中每一个实例都必须使用相同的名称
org.quartz.scheduler.instanceName: DefaultQuartzScheduler
#远程管理相关的配置,全部关闭
org.quartz.scheduler.rmi.export: false
org.quartz.scheduler.rmi.proxy: false
org.quartz.scheduler.wrapJobExecutionInUserTransaction: false
ThreadPool 实现的类名
org.quartz.threadPool.class: org.quartz.simpl.SimpleThreadPool
#线程数量
org.quartz.threadPool.threadCount: 10
#线程优先级 
org.quartz.threadPool.threadPriority: 5
#自创建父线程
org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread: true
#容许的最大作业延
org.quartz.jobStore.misfireThreshold: 60000
#ID设置为自动获取 每一个必须不同 
org.quartz.scheduler.instanceId: AUTO
#数据保存方式为持久化
org.quartz.jobStore.class: org.quartz.impl.jdbcjobstore.JobStoreTX
#加入集群
org.quartz.jobStore.isClustered: true
#调度实例失效的检查时间间隔
org.quartz.jobStore.clusterCheckinInterval: 10000

1.1.3 在项目中加入Quartz的初始化信息: 命名spring-quartz.xml

<?xml version="1.0" encoding="UTF-8"?><beans       xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"><bean id="quartzScheduler"        class="org.springframework.scheduling.quartz.SchedulerFactoryBean"><!-- 自定义的bean注入类,解决job里面无法注入spring的service的问题 --><property name="jobFactory"><bean class="com.fc.sales.control.statistics.job.SpringBeanJobFactory" /></property><!-- quartz的数据源 --><property name="dataSource" ref="quartz" /><!-- quartz的基本配置信息引入 --><property name="configLocation" value="classpath:quartz.properties"/>
<!-- 调度标识名 --><property name="schedulerName" value="DefaultQuartzScheduler" /><!--必须的,QuartzScheduler 延时启动,应用启动完后 QuartzScheduler 再启动 --><property name="startupDelay" value="30" /><!-- 通过applicationContextSchedulerContextKey属性配置spring上下文 --><property name="applicationContextSchedulerContextKey" value="applicationContextKey" /><!--可选,QuartzScheduler 启动时更新己存在的Job,这样就不用每次修改targetObject后删除qrtz_job_details表对应记录了 --><property name="overwriteExistingJobs" value="true" /><!-- 设置自动启动 --><property name="autoStartup" value="true" /><!-- 注册触发器 --><property  name="triggers"><list><ref bean="orderSyncScannerTrigger" /></list></property><!-- 注册jobDetail --><property name="jobDetails"><list><ref bean="orderSyncDetail" /></list></property></bean>  <!--配置调度具体执行的方法-->  <bean id="orderSyncDetail"  class="org.springframework.scheduling.quartz.JobDetailFactoryBean">  <property name="jobClass" value="com.fc.sales.control.statistics.job.OrderSyncJob"/><property name="durability" value="true" />    <property name="requestsRecovery" value="true" /> </bean> <!--配置调度执行的触发的时间-->  <bean id="orderSyncScannerTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean">  <property name="jobDetail" ref="orderSyncDetail" />  <property name="cronExpression">  <!-- 每天上午00:30点执行任务调度 -->  <value>0 30 00 * * ?</value>  </property>  </bean>
  • 1.1.4 在web.xml启动项中加入spring-quartz.xml文件*
 <servlet><servlet-name>dispatcherServlet</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:/spring/spring-quartz.xml</param-value></init-param><load-on-startup>1</load-on-startup></servlet>
  • 1.1.5 附上对应的解决无法注入的jobFactory的代码*:

import org.quartz.spi.TriggerFiredBundle;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.config.AutowireCapableBeanFactory;
import org.springframework.scheduling.quartz.AdaptableJobFactory;
import org.springframework.stereotype.Component;/*** Created by lyndon on 16/9/13.*/
@Component
public class jobFactory extends AdaptableJobFactory {//这个对象Spring会帮我们自动注入进来,也属于Spring技术范畴.@Autowiredprivate AutowireCapableBeanFactory capableBeanFactory;protected Object createJobInstance(TriggerFiredBundle bundle) throws Exception {//调用父类的方法Object jobInstance = super.createJobInstance(bundle);//进行注入,这属于Spring的技术,不清楚的可以查看Spring的API.capableBeanFactory.autowireBean(jobInstance);return jobInstance;}
}

****1.2 quartz框架实现分布式定时任务的原理****;
Quartz集群中每个节点都是一个单独的Quartz应用,它又管理着其他的节点。这个集群需要每个节点单独的启动或停止;和我们的应用服务器集群不同,独立的Quratz节点之间是不需要 通信的。不同节点之间是通过数据库表来感知另一个应用。只有使用持久的JobStore才能完成Quartz集群。

untitled21.jpg

  • 1.2.1 既然Quartz分布式集群是利用数据库锁机制来实现集群环境下的并发控制,我们就需要了解Quratz的数据库表:可以去官方现在对于版本的sql文件导入。

untitled22.png

  • 1.2.2 Quartz线程模型:
    Quartz中有两类线程:Scheduler调度线程和任务执行线程。

  • 任务执行线程: Quartz不会在主线程(QuartzSchedulerThread)中处理用户job。Quratz是将线程管理的职责委托给ThreadPool,一般的设置使用SimpleThreadPool,SimpleThreadPool创建一定数量的工作线程(WorkerThread),当然这样就意味所有的线程都是异步操作的,所以我们在工作线程的job里面实现业务的时候是没必要重新去创建一个新的线程的,在Quartz创建工作线程的时候已经完成了异步任务的创建。

  • Scheduler调度线程:QuartzScheduler被创建的时候会创建一个QuratzSchedulerThread实例。

  • 1.2.3 Quartz源码分析:

  • QuartzSchedulerThreand包含有决定何时下一个Job将被触发的处理循环,主要的逻辑在其的run()方法中,如下图:

untitled23.png

由此可知,QuartzSchedulerThread不断的在获取trigger,触发trigger,释放trigger。
那么具体又是如何获取trigger的呢,可以从上面的源码中可以发现:qsRsrcs.getJobStore()返回对象是JobStore ,具体的集群配置参考1.1.2. org.quartz.jobStore.class:org.quartz.impl.jdbcjobstore.JobStoreTX
JobStoreTx继承自JobStoreSupport,而JobStoreSupport的acquireNextTrigger,triggerFired,releaseAcquiredTrigger方法负责具体trigger相关操作,都必须获得TRIGGER-ACCESS锁。核心逻辑在executeInNonManagedTxLock方法中。

untitled24.png

由上代码可知Quartz集群基于数据库锁的同步操作流程如下图所示:

untitled25.png



/

 

Quartz分布式定时任务的暂停和恢复等

前两篇我们了解了quartz分布式定时任务的基本原理和实现方式,知道所有的定时任务都会被持久化到数据库。那么我们肯定可以通过操作数据库来做定时任务的暂停,恢复,立即启动,添加等操作。
事实上,quartz已经给我们提供来一些列的api接口来操作对应的定时任务,我们只需要在这个基础之上做进一步的扩展和封装就可以实现我们自己业务,下面,将围绕定时任务的控制,提供一个简单的实现方式。

使用的环境版本:spring4.x+quartz2.2.x

1,首先,我们需要创建一个我们自己job的实体类ScheduleJob:

/** 
* Created by lyndon on 16/9/13. * job的实体类*/
public class ScheduleJob {   private String jobNo; //任务编号    private String jobName; //任务名称    private String jobGroup; //任务所属组    private String desc; //任务描述          private String jobStatus; //任务状态    private String cronExpression; //任务对应的时间表达式private String triggerName; //触发器名称//此处省略get和set方法}

2, 创建我们自己的QuartzImplService服务层:

import org.quartz.*;
import org.quartz.impl.matchers.GroupMatcher;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;import javax.annotation.Resource;
import java.util.*;/*** Created by lyndon on 16/9/13.* quartz_job的工具类*/
@Service
public class QuartzUtils {private final Logger logger = LoggerFactory.getLogger(QuartzUtils.class);@Resourceprivate Scheduler scheduler;/**** 获取计划任务列表* @return  List<ScheduleJob>*/public List<ScheduleJob> getPlanJobList() throws SchedulerException{List<ScheduleJob> jobList = new ArrayList<>();GroupMatcher<JobKey> matcher = GroupMatcher.anyJobGroup();Set<JobKey> jobKeys = scheduler.getJobKeys(matcher);;jobKeys = scheduler.getJobKeys(matcher);for (JobKey jobKey : jobKeys) {List<? extends Trigger> triggers = scheduler.getTriggersOfJob(jobKey);for (Trigger trigger : triggers) {ScheduleJob job = new ScheduleJob();job.setJobName(jobKey.getName());job.setJobGroup(jobKey.getGroup());// 此处是我自己业务需要,给每个定时任务配置类对应的编号和描述String value = PropertiesUtils.getStringCN(jobKey.getName());if(null != value && !"".equals(value)){job.setJobNo(value.split("/")[0]);job.setDesc(value.split("/")[1]);}else{job.setJobNo("0000");job.setDesc("未监控任务");}job.setTriggerName("触发器:" + trigger.getKey());Trigger.TriggerState triggerState = scheduler.getTriggerState(trigger.getKey());job.setJobStatus(triggerState.name());if (trigger instanceof CronTrigger) {CronTrigger cronTrigger = (CronTrigger) trigger;String cronExpression = cronTrigger.getCronExpression();job.setCronExpression(cronExpression);}jobList.add(job);}}// 对返回的定时任务安装编号做排序Collections.sort(jobList,new Comparator<ScheduleJob>(){public int compare(ScheduleJob arg0, ScheduleJob arg1) {return arg0.getJobNo().compareTo(arg1.getJobNo());}});return jobList;}/*** 获取正在运行的任务列表* @return List<ScheduleJob>*/public List<ScheduleJob> getCurrentJobList() throws SchedulerException{List<JobExecutionContext> executingJobs = scheduler.getCurrentlyExecutingJobs();;List<ScheduleJob> jobList = new ArrayList<ScheduleJob>(executingJobs.size());;for (JobExecutionContext executingJob : executingJobs) {ScheduleJob job = new ScheduleJob();JobDetail jobDetail = executingJob.getJobDetail();JobKey jobKey = jobDetail.getKey();Trigger trigger = executingJob.getTrigger();job.setJobName(jobKey.getName());job.setJobGroup(jobKey.getGroup());String value = PropertiesUtils.getStringCN(jobKey.getName());if(null != value && !"".equals(value)){job.setJobNo(value.split("/")[0]);job.setDesc(value.split("/")[1]);}else{job.setJobNo("0000");job.setDesc("未监控任务");}job.setTriggerName("触发器:" + trigger.getKey());Trigger.TriggerState triggerState = scheduler.getTriggerState(trigger.getKey());job.setJobStatus(triggerState.name());if (trigger instanceof CronTrigger) {CronTrigger cronTrigger = (CronTrigger) trigger;String cronExpression = cronTrigger.getCronExpression();job.setCronExpression(cronExpression);}jobList.add(job);}Collections.sort(jobList,new Comparator<ScheduleJob>(){public int compare(ScheduleJob arg0, ScheduleJob arg1) {return arg0.getJobNo().compareTo(arg1.getJobNo());}});return  jobList;}/*** 暂停当前任务* @param scheduleJob*/public void pauseJob(ScheduleJob scheduleJob) throws SchedulerException{JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());if(scheduler.checkExists(jobKey)){scheduler.pauseJob(jobKey);}}/*** 恢复当前任务* @param scheduleJob*/public void resumeJob(ScheduleJob scheduleJob) throws SchedulerException{JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());if(scheduler.checkExists(jobKey)){//并恢复scheduler.resumeJob(jobKey);//重置当前时间this.rescheduleJob(scheduleJob);}}/*** 删除任务* @param scheduleJob* @return boolean*/public boolean deleteJob(ScheduleJob scheduleJob) throws SchedulerException{JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());if(scheduler.checkExists(jobKey)){return scheduler.deleteJob(jobKey);}return false;}/*** 立即触发当前任务* @param scheduleJob*/public void triggerJob(ScheduleJob scheduleJob) throws SchedulerException{JobKey jobKey = JobKey.jobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());if(scheduler.checkExists(jobKey)){scheduler.triggerJob(jobKey);}}/*** 更新任务的时间表达式* @param scheduleJob* @return Date*/public Date rescheduleJob(ScheduleJob scheduleJob) throws SchedulerException{TriggerKey triggerKey = TriggerKey.triggerKey(scheduleJob.getJobName(),scheduleJob.getJobGroup());if(scheduler.checkExists(triggerKey)){CronTrigger trigger = (CronTrigger) scheduler.getTrigger(triggerKey);CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule(scheduleJob.getCronExpression());//按新的cronExpression表达式重新构建triggertrigger = trigger.getTriggerBuilder().withIdentity(triggerKey).withSchedule(scheduleBuilder).build();//按新的trigger重新设置job执行return scheduler.rescheduleJob(triggerKey, trigger);}return null;}/*** 查询其中一个任务的状态* @param scheduleJob* @return* @throws SchedulerException*/public String scheduleJob(ScheduleJob scheduleJob) throws SchedulerException {String status = null;TriggerKey triggerKey = TriggerKey.triggerKey(scheduleJob.getJobName(), scheduleJob.getJobGroup());CronTrigger trigger = (CronTrigger) scheduler.getTrigger(triggerKey);if (null != trigger) {Trigger.TriggerState triggerState = scheduler.getTriggerState(trigger.getKey());status = triggerState.name();}return status;}/*** 校验job是否已经加载* @param scheduleJob  JOB基本信息参数* @return          是否已经加载*/public boolean checkJobExisted(ScheduleJob scheduleJob) throws SchedulerException {return scheduler.checkExists(new JobKey(scheduleJob.getJobName(), scheduleJob.getJobGroup()));}private String getStatuDesc(String status){if(status.equalsIgnoreCase("NORMAL")){return "正常";}else if(status.equalsIgnoreCase("PAUSED")){return "暂停";}else{return "异常";}}
}

3,提供对应的Controller


import com.innmall.hotelmanager.common.Result;
import com.innmall.hotelmanager.service.quartz.QuartzUtils;
import com.innmall.hotelmanager.service.quartz.ScheduleJob;
import org.quartz.SchedulerException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import javax.annotation.Resource;
import java.util.List;/*** Created by lyndon on 16/9/13.*/
@RestController
@RequestMapping(value = {"/v1/job"})
public class QuartzController {private Logger logger = LoggerFactory.getLogger(this.getClass());@Resourceprivate QuartzUtils quartzUtils;//获取定时任务的列表@RequestMapping(value = {"/getJobList"})public Result getPlanJobList(String openId){//QuartzUtils quartzUtils = new QuartzUtils();List<ScheduleJob> list = null;try {list = quartzUtils.getPlanJobList();} catch (SchedulerException e) {e.printStackTrace();}return Result.success(list);}//暂停任务@RequestMapping(value = {"/pauseJob"})public Result pauseJob(String openId){//QuartzUtils quartzUtils = new QuartzUtils();ScheduleJob job = new ScheduleJob();job.setJobGroup("innmall_job");job.setJobName("refreshWxToKenJobDetail");try {quartzUtils.pauseJob(job);} catch (SchedulerException e) {e.printStackTrace();}return Result.success("暂停成功");}//恢复任务@RequestMapping(value = {"/resumeJob"})public Result resumeJob(String openId){//QuartzUtils quartzUtils = new QuartzUtils();ScheduleJob job = new ScheduleJob();job.setJobGroup("innmall_job");job.setJobName("refreshWxToKenJobDetail");try {quartzUtils.resumeJob(job);} catch (SchedulerException e) {e.printStackTrace();}return Result.success("恢复成功");}//立即触发任务@RequestMapping(value = {"/triggerJob"})public Result triggerJob(String openId){//QuartzUtils quartzUtils = new QuartzUtils();ScheduleJob job = new ScheduleJob();job.setJobGroup("innmall_job");job.setJobName("refreshWxToKenJobDetail");try {quartzUtils.triggerJob(job);} catch (SchedulerException e) {e.printStackTrace();}return Result.success("触发成功");}//删除任务@RequestMapping(value = {"/deleteJob"})public Result deleteJob(String openId){//QuartzUtils quartzUtils = new QuartzUtils();ScheduleJob job = new ScheduleJob();job.setJobGroup("innmall_job");job.setJobName("refreshWxToKenJobDetail");try {quartzUtils.deleteJob(job);} catch (SchedulerException e) {e.printStackTrace();}return Result.success("触发成功");}}

4,接下来,我们就可以进行单元测试了。

5,需要注意的地方:

5.1 service层:

@Resource
private Scheduler scheduler;

这里是因为我们在xml里面已经配置对应的工厂bean,所以可以在这里可以直接注入:

<bean id="quartzScheduler" class="org.springframework.scheduling.quartz.SchedulerFactoryBean"     destroy-method="destroy">

5.2 关于区分不同业务的触发器和任务,可以配置job和trigger的group属性,这样我们便以区分,如果不设置,quartz将使用default关键字:

   <bean id="refreshWxToKenJobDetail"class="org.springframework.scheduling.quartz.JobDetailFactoryBean"><property name="jobClass" value="com.innmall.hotelmanager.timer.RefreshWxToKen"/><property name="durability" value="true" /><property name="requestsRecovery" value="true" /><property name="group" value="innmall_job"/></bean><bean id="refreshWxToKenTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean"><property name="jobDetail" ref="refreshWxToKenJobDetail"/><!-- 每10s钟运行一次 --><property name="cronExpression" value="0/10 * * * * ?"/><property name="misfireInstruction" value="2"/><property name="group" value="innmall_trigger"/></bean>

5.3 关于定时任务恢复后,我们如果不需要让之前错过的定时任务再执行一次,可以设置misfireInstruction的属性,其实就是

CronTrigger.MISFIRE_INSTRUCTION_DO_NOTHING

进去可以看见对应的值为2.
并且需要在我们恢复任务的时候调用更新的方法,可以见上文的QuartzUtil中的方法。

//重置当前时间
this.rescheduleJob(scheduleJob);

5.4 如果需要定时任务恢复后,需要将之前错过的执行一次,那么只需要在xml里面去除misfireInstruction属性,其实就是使用默认配置,并且在恢复的时候不调用更新的方法。

关于quartz的使用方法,暂时就介绍到这里,如果有什么地方有问题,欢迎指正,后面将持续研究对应的异常处理机制,敬请关注~


 


http://chatgpt.dhexx.cn/article/4AOswMEi.shtml

相关文章

python类型转换函数str

str函数&#xff0c;将数字转为字符串&#xff1a;

Python 类型转换(数据类型转换函数大全)

文章目录 虽然 Python 是弱类型编程语言&#xff0c;不需要像 Java 或 C 语言那样还要在使用变量前声明变量的类型&#xff0c;但在一些特定场景中&#xff0c;仍然需要用到类型转换。 比如说&#xff0c;我们想通过使用 print() 函数输出信息“您的身高&#xff1a;”以及浮点…

python怎么转换文件格式_python怎么转换数据类型

在处理数据的时候&#xff0c;经常需要转换数据的格式&#xff0c;来方便数据遍历等操作。下面我们来看一下Python中的几种数据类型转换。 1、字符串转字典&#xff1a;dict_string "{name:linux,age:18}" to_dict eval(dict_string) print(type(to_dict)) 也可以用…

python强制类型转换

1.强转为int string->int a string print(int(a))a 1.2 print(int(a))a 12 print(int(a))string仅在无特殊字符&#xff08;包括小数点&#xff09;且全为数字的情况下可强转为float boolen->int a True b False print(int(a)) print(int(b))float->int a …

python批量转换数据类型_python中数据类型转换

1、list转str 假设有一个名为test_list的list,转换后的str名为test_str 则转换方法: test_str = "".join(test_list) 例子: 需要注意的是该方法需要list中的元素为字符型,若是整型,则需要先转换为字符型后再转为str类型。 2、str转list 假设有一个名为test_str的…

python类型转换

一、int——支持转换为 int 类型的&#xff0c;仅有 float、str、bytes&#xff0c;其他类型均不支持。 1、str与bytes类型是什么&#xff0c;有什么区别 文本总是Unicode&#xff0c;由str类型表示&#xff0c;二进制数据则由bytes类型表示。 字符串是 以字符为单位进行处理…

浅谈Python中的类型转换

目录 &#xff08;一&#xff09;前言 &#xff08;二&#xff09;四种常见转换类型 1. int()函数 &#xff08;1&#xff09;int()函数的格式 &#xff08;2&#xff09;示例 2. float()函数 &#xff08;1&#xff09;float()函数格式&#xff1a; &#xff08;2&…

初学ansys:模态分析及谐响应分析

谐响应为线性系统对简谐激励的稳态响应&#xff0c;当系统含有阻尼或者激励为复数&#xff08;相位不为0或pi&#xff09;,谐响应为复数。 ansys可在模态分析的基础上进行谐响应分析&#xff1a; 2阶频率377.47&#xff0c;振型如下&#xff1a; 当z向加速度10g,右端平面z向位…

关于模态分析

模态分析目的是获得固有频率、模态振型、振型参与系数、有效质量 模态分析是动力学的基础分析,谐响应分析的前提是进行模态分析。 什么是固有频率呢&#xff1f;共振频率呢&#xff1f; 比如一个单摆做好后&#xff0c;他的振动频率等于2Pi&#xff08;l/g&#xff09;^&#x…

Ansys-模态分析基础上的谱分析学习收获

谱分析是一种将模态分析和已知谱联系起来的、计算结构位移和应力的分析方法&#xff0c;主要用于确定结构对随机载荷或时间变化载荷&#xff08;如地震载荷&#xff09;的动力响应。谱是谱值和频率的关系曲线&#xff0c;它反映了时间历程载荷的强度和频率之间的关系。响应谱&a…

什么是模态分析?什么是振型?

模态和振型是两个比较难懂的概念&#xff0c;涉及的理论比较多&#xff0c;我想通过一句话引出&#xff0c;然后通过逐步解释的方法去阐释这两个概念。 以一根梁为例&#xff0c;通过理论计算寻找其固有频率、阻尼比、振型的过程就是解析模态分析&#xff0c;通过实验得到的就…

模态分析实例—斜齿圆柱齿轮的固有频率分析

本例介绍了对一个复杂结构—斜齿圆柱齿轮模型的创建方法&#xff0c;以及利用ANSYS对其进行固有频率和振型研究即模态分析的方法、步骤和过程。 APDL: /CLEAR,NOSTART /FILNAME,EXAMPLE11/PREP7 ET,1,SOLID45 MP,EX,1,2E11 MP,PRXY,1,0.3 MP,DENS,1,7800 K,1,21.87E-3 K,2,22.…

使用ANSYS进行对称边界的模态分析,制作【春节快乐】

这里写自定义目录标题 想法由来1. 建模2. 使用对称边界进行模态分析2.1 处理几何模型2.2 网格划分并设置边界条件进行计算2.3 求解2.4查看结果 延续去年的传统&#xff0c;最近几天一直在想做个什么东西来迎接新年。本来想用keras训练个深度网络&#xff0c;从一大堆图片中识别…

[Ansys Workbench] 平面对称斜拉桥的模态分析

1. 题目 2. 预处理 使用静态结构和模态分析两个模块 2.1 定义材料 2.2 定义几何结构 使用 DesignModeler 不知道 DM 中怎么使用对称轴画图……我就用了笨方法画了 一个主梁 使用 Concept - Lines From Sketches 从草图生成线 得到的线在结构树中如图所示 选择草图中所有的线&…

Ansys模态计算结果图片批量导出命令流

1.设置图片输出质量 2.命令流一键导出 *do,i,1,10 ###输出前10阶计算结果图片 set,1,i ####构造循环 plnsol,u,sum /image,save,D:\ANSYS\ban%i%,bmp ### 输出文件路径 *enddo

ANSYS apdl命令流笔记15-------模态分析

模态分析简介 前言一、案例1--均匀直杆的固有频率分析(1)完整命令(2)固有频率(3)第2阶固有频率下的振型 二、案例2--有预应力的固定弦1.完整命令2.固有频率3.第2阶固有频率下的振型 前言 模态分析的本质就是研究系统的自由振动特性&#xff0c;确定一个结构的固有频率和振型。…

ANSYS_APDL——实例002-模态分析

/clear /prep7 et,1,link1 mp,ex,1,2e11 !材料属性 mp,prxy,1,0.3 mp,dens,1,7800 R,1,1e-6 !定义元素实常数 k,1,0,0,0 !定义1的坐标为000 k,2,1,0,0 !定义2的坐标为100 lstr,1,2 !定义点1到点2的直线 lesize,1,,,50 !线1&#xff0c;中间50结 lmesh,1 !沿着线生成节点…

ANSYS模态分析详细步骤记录

1&#xff0c;打开WorkBench&#xff0c;双击Modal&#xff0c;生成右边的表格 2&#xff0c; 右击Geometry&#xff0c;import一个模型&#xff0c;我这里已经添加了就变成了另一个选项 3&#xff0c;双击Model&#xff0c;打开开始编辑 4&#xff0c;打开之后选择材料&#…

基于ANSYS的无约束梁的模态分析与实验结果比较

一、实验模型简介 该模型来源于文献&#xff1a;“Khatir, A., Capozucca, R., Khatir, S. et al. Vibration-based crack prediction on a beam model using hybrid butterfly optimization algorithm with artificial neural network. Front. Struct. Civ. Eng. 16, 976–98…