scheduler定时调度系统是大多行业项目都需要的,传统的spring-job模式,个人感觉已经out了,因为存在很多的问题,特别是定时调度的追加、修改、删除等,需要修改xml,xml的配置生效无非是热部署灰度发布方案或者直接停止、重启服务器,完全不能做到自动启动、修复方式。
提醒:可以对应用进行集群部署,在对定时调度配置时可以使用集群方式或者单边配置应用方式,今天讲解的是使用spring4+scheduler实现定时调度,闲话少说,直接把步骤记录下来:
1. 在项目的pom.xml文件中引入quartz的jar包,如下:
<!-- quartz定时调度 --> lt;dependency> <groupId>org.quartz-scheduler</groupId> <artifactId>quartz</artifactId> <version>1.8.5</version> lt;/dependency>
2. 定义quartz的配置文件spring-context-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:context="http://www.springframework.org/schema/context" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd" default-lazy-init="false"> <!-- 调度器 --> <bean name="scheduler" class="org.springframework.scheduling.quartz.SchedulerFactoryBean"> <!-- 通过applicationContextSchedulerContextKey属性配置spring上下文 --> <property name="applicationContextSchedulerContextKey" value="applicationContext" /> </bean> <!--加载数据库任务--> <bean id="jobService" class="com.ml.honghu.job.service.JobService" init-method="loadJob" /> </beans>
3. 在项目的web.xml文件中引入spring-context-quartz.xml配置文件
classpath*:spring-context-quartz.xml
4. 定义job实体对象
public class Job{ private static final long serialVersionUID = 1L; /** * 任务执行周期cron表达式 */ public static int EXECYCLE_CRON = 2; /** * 任务执行周期自定义 */ public static int EXECYCLE_DEFINE = 1; /** * 执行周期-分钟 */ public static int EXECYCLE_MINUTE = 1; /** * 执行周期-小时 */ public static int EXECYCLE_HOUR = 2; /** * 执行周期-日 */ public static int EXECYCLE_DAY = 3; /** * 执行周期-月 */ public static int EXECYCLE_WEEK = 4; /** * 执行周期-月 */ public static int EXECYCLE_MONTH = 5; private String jobType; // 任务类型(1首页静态化、2栏目页静态化、3内容页静态化、4采集、5分发) private String jobName; // 任务名称 private String jobClass; // 任务类 private String execycle; // 执行周期分类(1非表达式 2 cron表达式) private String dayOfMonth; // 每月的哪天 private String dayOfWeek; // 周几 private String hour; // 小时 private String minute; // 分钟 private String intervalHour; // 间隔小时 private String intervalMinute; // 间隔分钟 private String jobIntervalUnit; // 1分钟、2小时、3日、4周、5月 private String cronExpression; // 规则表达式 private String isEnable; // 是否启用 public Job() { super(); } public Job(String id){ super(id); } @Length(min=1, max=1, message="任务类型(1首页静态化、2栏目页静态化、3内容页静态化、4采集、5分发)长度必须介于 1 和 1 之间") public String getJobType() { return jobType; } public void setJobType(String jobType) { this.jobType = jobType; } @Length(min=1, max=255, message="任务名称长度必须介于 1 和 255 之间") public String getJobName() { return jobName; } public void setJobName(String jobName) { this.jobName = jobName; } @Length(min=1, max=255, message="任务类长度必须介于 1 和 255 之间") public String getJobClass() { return jobClass; } public void setJobClass(String jobClass) { this.jobClass = jobClass; } @Length(min=1, max=1, message="执行周期分类(1非表达式 2 cron表达式)长度必须介于 1 和 1 之间") public String getExecycle() { return execycle; } public void setExecycle(String execycle) { this.execycle = execycle; } @Length(min=0, max=11, message="每月的哪天长度必须介于 0 和 11 之间") public String getDayOfMonth() { return dayOfMonth; } public void setDayOfMonth(String dayOfMonth) { this.dayOfMonth = dayOfMonth; } @Length(min=0, max=1, message="周几长度必须介于 0 和 1 之间") public String getDayOfWeek() { return dayOfWeek; } public void setDayOfWeek(String dayOfWeek) { this.dayOfWeek = dayOfWeek; } @Length(min=0, max=11, message="小时长度必须介于 0 和 11 之间") public String getHour() { return hour; } public void setHour(String hour) { this.hour = hour; } @Length(min=0, max=11, message="分钟长度必须介于 0 和 11 之间") public String getMinute() { return minute; } public void setMinute(String minute) { this.minute = minute; } @Length(min=0, max=11, message="间隔小时长度必须介于 0 和 11 之间") public String getIntervalHour() { return intervalHour; } public void setIntervalHour(String intervalHour) { this.intervalHour = intervalHour; } @Length(min=0, max=11, message="间隔分钟长度必须介于 0 和 11 之间") public String getIntervalMinute() { return intervalMinute; } public void setIntervalMinute(String intervalMinute) { this.intervalMinute = intervalMinute; } @Length(min=0, max=1, message="1分钟、2小时、3日、4周、5月长度必须介于 0 和 1 之间") public String getJobIntervalUnit() { return jobIntervalUnit; } public void setJobIntervalUnit(String jobIntervalUnit) { this.jobIntervalUnit = jobIntervalUnit; } @Length(min=0, max=255, message="规则表达式长度必须介于 0 和 255 之间") public String getCronExpression() { return cronExpression; } public void setCronExpression(String cronExpression) { this.cronExpression = cronExpression; } @Length(min=1, max=1, message="是否启用长度必须介于 1 和 1 之间") public String getIsEnable() { return isEnable; } public void setIsEnable(String isEnable) { this.isEnable = isEnable; } }
5. 编写quartz的jobServvice类:
package com.ml.honghu.job.service; import java.text.ParseException; import java.util.List; import java.util.UUID; import org.quartz.CronTrigger; import org.quartz.JobDetail; import org.quartz.Scheduler; import org.quartz.SchedulerException; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import com.ml.honghu.StringUtils; import com.ml.honghu.common.persistence.Page; import com.ml.honghu.common.service.CrudService; import com.ml.honghu.job.dao.JobDao; import com.ml.honghu.job.entity.Job; /** * 定时调度任务Service * * @author honghu */ @Service @Transactional(readOnly = true) public class JobService extends CrudService<JobDao, Job> { @Autowired private JobDao jobDao; private Logger logger = LoggerFactory.getLogger(getClass()); public Job get(String id) { return super.get(id); } public List<Job> findList(Job job) { return super.findList(job); } public Page<Job> findPage(Page<Job> page, Job job) { return super.findPage(page, job); } @Transactional(readOnly = false) public void save(Job job) { super.save(job); // 启用则启动任务 if (StringUtils.equals("1", job.getIsEnable())) { startTask(job, job.getId()); } } @Transactional(readOnly = false) public void update(Job job) { //结束定时调度 endTask(job.getId()); job.preUpdate(); jobDao.update(job); // 启用则启动任务 if (StringUtils.equals("1", job.getIsEnable())) { startTask(job, job.getId()); } } @Transactional(readOnly = false) public void delete(Job job) { //结束任务 endTask(job.getId()); super.delete(job); } /** * 系统初始加载任务 */ public void loadJob() throws Exception { List<Job> jobList = this.findList(new Job()); if (null != jobList && jobList.size() > 0) { for (int i = 0; i < jobList.size(); i++) { Job job = jobList.get(i); // 任务开启状态 执行任务调度 if (StringUtils.equals("1", job.getIsEnable())) { try { JobDetail jobDetail = new JobDetail(); // 设置任务名称 if (StringUtils.isNotBlank(job.getId())) { jobDetail.setName(job.getId()); } else { UUID uuid = UUID.randomUUID(); jobDetail.setName(uuid.toString()); job.setId(uuid.toString()); } jobDetail.setGroup(Scheduler.DEFAULT_GROUP); // 设置任务执行类 jobDetail.setJobClass(getClassByTask(job.getJobClass())); // 添加任务参数 CronTrigger cronTrigger = new CronTrigger("cron_" + i, Scheduler.DEFAULT_GROUP, jobDetail.getName(), Scheduler.DEFAULT_GROUP); cronTrigger.setCronExpression(getCronExpressionFromDB(job.getId())); // 调度任务 scheduler.scheduleJob(jobDetail, cronTrigger); } catch (SchedulerException e) { logger.error("JobService SchedulerException", e); } catch (ClassNotFoundException e) { logger.error("JobService ClassNotFoundException", e); } catch (Exception e) { logger.error("JobService Exception", e); } } } } } /** * * @param taskClassName * 任务执行类名 * @return * @throws ClassNotFoundException */ @SuppressWarnings("rawtypes") private Class getClassByTask(String taskClassName) throws ClassNotFoundException { return Class.forName(taskClassName); } public String getCronExpressionFromDB(String id) throws Exception { // 设置任务规则 Job job = this.get(id); if (null != job) { if (Job.EXECYCLE_CRON == Integer.parseInt(job.getExecycle())) { return job.getCronExpression(); } else { Integer execycle = Integer.parseInt(job.getJobIntervalUnit()); String excep = ""; if (execycle.equals(Job.EXECYCLE_MONTH)) { excep = "0 " + job.getMinute() + " " + job.getHour() + " " + job.getDayOfMonth() + " * ?"; } else if (execycle.equals(Job.EXECYCLE_WEEK)) { excep = "0 " + job.getMinute() + " " + job.getHour() + " " + " ? " + " * " + job.getDayOfWeek(); } else if (execycle.equals(Job.EXECYCLE_DAY)) { excep = "0 " + job.getMinute() + " " + job.getHour() + " " + " * * ?"; } else if (execycle.equals(Job.EXECYCLE_HOUR)) { excep = "0 0 */" + job.getIntervalHour() + " * * ?"; } else if (execycle.equals(Job.EXECYCLE_MINUTE)) { excep = "0 */" + job.getIntervalMinute() + " * * * ?"; } return excep; } } return ""; } private void startTask(Job job, String id) { try { String cronExpress = getCronExpressionFromDB(id); if (StringUtils.isNotEmpty(cronExpress) && cronExpress.indexOf("null") == -1) { JobDetail jobDetail = new JobDetail(); jobDetail.setName(id); jobDetail.setGroup(Scheduler.DEFAULT_GROUP); jobDetail.setJobClass(getClassByTask(job.getJobClass())); CronTrigger cronTrigger = new CronTrigger("cron_" + id, Scheduler.DEFAULT_GROUP, jobDetail.getName(), Scheduler.DEFAULT_GROUP); cronTrigger.setCronExpression(cronExpress); scheduler.scheduleJob(jobDetail, cronTrigger); } } catch (ParseException e) { logger.error("JobService ParseException", e); } catch (Exception e) { logger.error("JobService Exception", e); } } private void endTask(String id) { try { scheduler.deleteJob(id, Scheduler.DEFAULT_GROUP); } catch (SchedulerException e) { logger.error("JobService endTask", e); } } @Autowired private Scheduler scheduler; }
6. 编写相关job的Controller、dao、dao.xml我这边就不写了,其实就是对数据的增删改查操作
7. 启动项目验证quartz是否成功:
项目启动个控制台:
任务列表:
任务添加和修改界面:
到此完毕!(企业架构源码可以加求球:三五三六二四七二五九)
相关推荐
在Go语言中,实现多线程定时任务是一个常见的需求,特别是在服务器端编程或者后台任务处理中。Go语言的并发模型是基于CSP(Communicating Sequential Processes)理论的,通过goroutine和channel来实现轻量级线程和...
在IT行业中,多线程定时任务邮件服务是一个常见的需求,特别是在服务器管理和自动化运维场景下。这个服务能够定期执行某些任务,并通过电子邮件的方式将结果或通知发送给相关人员。下面,我们将详细探讨这一主题。 ...
在IT行业中,多线程定时任务邮件服务是一种常见的系统功能,它主要用于自动化处理,例如监控、报警或定期报告。此服务结合了多线程技术、定时任务调度以及邮件通信,通常用于大型分布式系统中,以确保高效且及时地...
本篇文章将详细解释如何利用注解优雅地实现多线程定时任务。 首先,我们需要理解“定时任务”。定时任务是一种能够按照预设的时间间隔或特定时间点执行的程序功能。在Java中,我们可以使用`java.util.Timer`和`java...
- **集群部署**: 调度中心和执行器都可以集群部署,调度中心的集群通过数据库的分布式锁保证调度的一致性,执行器集群则通过心跳机制动态感知任务分发。 - **负载均衡**: 调度中心根据执行器的负载情况,智能选择...
综上所述,这个基于QT的多线程定时器项目展示了如何在QT环境下有效地使用多线程和定时器,提高程序性能并实现复杂的时间调度。开发者通过学习这个项目,可以掌握如何在QT中设计用户界面、管理线程、使用定时器,以及...
定时器与多线程任务调度器使用定时器与多线程任务调度器使用
在.NET框架中,C#提供了两种Timer类,用于在多线程环境中实现定时触发操作:System.Threading.Timer和System.Timers.Timer。这两种Timer虽然都与时间调度相关,但在使用上和特性上有所不同。在这个主题中,我们将...
Java多线程是Java编程中的重要概念,尤其在如今的多核处理器环境下,理解并熟练掌握多线程技术对于提高程序性能和响应速度至关重要。本资料详细讲解了Java多线程的原理,并提供了丰富的实战代码,非常适合Java初学者...
在实际开发中,你可能还需要考虑多线程、异常处理以及任务取消等复杂情况。 总之,监听器定时调度是Java中实现定时任务的一种有效方式,适用于那些需要在后台定期执行的逻辑。通过熟练掌握 `Timer` 和 `TimerTask`...
总之,“桌面精灵应用多线程滚动字幕万年历定时提醒等技术”是一个综合性的项目,涵盖了Java编程、用户界面设计、日期处理以及任务调度等多个方面的知识。通过合理运用这些技术,可以构建出一款功能丰富、用户体验...
- 使用线程安全的组件,特别是在多线程环境下运行的Job。 - 对于长时间运行的任务,考虑使用异步处理或单独的服务。 通过这个"quartz定时调度完整实例",你可以学习到如何在实际项目中利用Quartz来自动化执行各种...
Java中的动画可以通过多线程控制图像的定时更新来实现。 11. **多线程动画应用**:利用多线程实现动画,可独立控制每个元素的运动,增强用户体验,提高程序响应速度。 #### 二、实验内容详解 1. **多线程程序设计*...
Quartz调度器采用多线程的方式,可以支持任务的并发执行,从而提高调度效率。Quartz的核心组件包括Scheduler接口、Job类和Trigger类。 - Scheduler接口负责管理调度任务的生命周期,可以启动、停止、暂停调度器。 -...
4. **并发安全**:gocron内部实现了并发控制,确保在多线程环境下安全执行任务。 5. **任务管理**:可以方便地暂停、恢复或删除已注册的任务,便于任务生命周期的管理。 6. **可扩展性**:通过钩子函数(hooks)机制...
Java多线程机制是Java编程中至关重要的一部分,它允许程序同时执行多个任务,提升应用程序的效率和响应性。以下是对各个知识点的详细说明: 9.1 Java中的线程: Java程序中的线程是在操作系统级别的线程基础上进行...
此外,文档可能会涵盖性能优化方面的内容,比如如何评估和调整任务执行效率,避免资源浪费,以及如何在多线程环境下确保任务调度的准确性和一致性。 最后,文档可能还会提供一些最佳实践和常见问题解答,帮助开发者...
此时,可以在定时任务中调用标记了`@Async`的方法,实现定时任务和多线程任务的完美结合。 总结,Spring 3.0 MVC中的定时器和多线程任务是提升应用程序效率的重要工具。定时器可以帮助我们定期执行特定任务,而多...
在这个“易语言-多线程定时关闭简单例子”中,我们将探讨如何在易语言中实现多线程以及定时器功能,这对于理解和应用易语言进行系统级操作或者自动化任务具有重要的学习价值。 首先,我们要理解什么是多线程。在...
- **线程安全**:jCrontab内部处理了多线程环境下的并发问题,确保任务调度的正确性。 - **事件通知**:提供了任务执行前后的回调接口,方便开发者进行日志记录、异常处理等操作。 **2. 使用jCrontab的步骤** - **...