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;
- }
相关推荐
本示例项目"Lucene4.8+IKAnalyzer+SpringMVC4+Jsoup+Quartz"为我们提供了一个强大的智能搜索引擎搭建框架,它整合了多项关键技术,旨在帮助开发者高效地实现网页抓取、内容分析和定时任务等功能。 首先,Lucene是...
在实际项目中,任务调度管理往往涉及到多线程和并发控制,确保任务的有序执行和资源的合理分配。开发者还需要关注异常处理,确保即使在任务执行过程中出现错误,系统也能保持稳定运行。 至于压缩包中的"Scientific...
SSM(Spring、SpringMVC、MyBatis)框架是Java Web开发中常见的技术栈,而Quartz是一款强大的任务调度库,常用于实现定时任务。本文将深入探讨如何在SSM项目中整合Quartz,并解决服务无法注入的问题。 首先,我们要...
在本文中,我们将详细介绍java多线程实现下载图片并压缩的过程,涵盖了 SpringMVC定时任务的实现、FTP环境搭建、图片下载和压缩等多个方面。 多线程实现下载图片 在多线程实现下载图片中,我们使用了SpringMVC定时...
但是Quartz有线程和线程管理以及集群等高级特性,所以大家可以自行选择了。不过一般情况下,觉得SpringTask足够了。 Spring Task提供两种方式进行配置,注解和配置文件。使用注解虽然简单,不用配置xml,但是相对于...
4. 文件上传、多线程下载服务化、发送邮件、短信服务化、部门信息服务化、产品信息服务化、信息发布服务化、我的订阅服务化、我的任务服务化、公共链接、我的收藏服务化等 系统模块: 1. 用户管理: 用户信息...
3. **异步处理**:为了提高爬虫效率,可以使用多线程或异步编程。Java 8引入了CompletableFuture,Spring 5提供了WebFlux框架,支持非阻塞和反应式编程。 **三、整合SpringMVC与Java爬虫** 1. **创建爬虫服务**:...
Java的多线程、面向对象特性使得处理并发请求和复杂业务逻辑变得更加简单。系统可能使用了Spring Boot来简化Java应用的配置和启动过程。 5. **微信小程序**:作为前端界面,微信小程序提供了轻量级、跨平台的用户...
包括JavaSE基础(多态、异常处理、常用API、数据类型、IO操作、集合、多线程和并发库、内部类)、JavaSE高级(反射、动态代理、设计模式&回收机制、加载器、JVM基础、GC基础)、JavaWeb基础(JDBC技术、HTTP协议、...
在多线程环境中,通常使用双重检查锁定(Double-Check Locking)来实现线程安全的单例。 - **策略模式**:定义一系列算法,封装起来并使它们可以互相替换,使算法的变化独立于使用算法的客户。 - **观察者模式**:...
开发者需要掌握面向对象编程、异常处理、多线程、集合框架等核心概念。 5. **数据库设计与操作**: - 在游戏助手小程序中,可能会涉及到用户信息、游戏数据、排行榜等功能,需要设计合理的数据库模型并使用SQL进行...
多线程编程是并发处理的核心,学习线程的创建、同步机制、线程池等知识将让你更好地处理高并发场景。 在项目管理工具方面,Maven是Java开发中的标准工具,用于管理依赖和构建项目。掌握Maven的基本操作和配置,可以...
他负责了业务模块的设计、权限管理、待办任务管理,用SpringMail发送提醒邮件,Quartz框架实现定时任务,POI技术处理数据导出,Redis用于菜单智能展示,而统计分析则可能借助amCharts等工具实现动态图表。...
他熟练掌握了Spring、SpringMVC和MyBatis等常用的Java后端框架,这些框架常用于构建企业级应用,Spring提供依赖注入和AOP(面向切面编程)等功能,SpringMVC则负责处理HTTP请求和响应,MyBatis则是轻量级的持久层...
1. 多线程基础知识:了解Java多线程编程的基础概念和技术。 2. 集合基础知识:掌握Java集合框架的基础知识和应用。 二、后台开发知识点: 1. Spring框架基础知识:熟悉Spring框架的基础概念和应用,包括依赖注入、...
开发者需熟悉Java语法、异常处理、多线程、集合框架等基础知识。 4. **SSM框架**: SSM(Spring、SpringMVC、MyBatis)是Java web开发中的经典组合。Spring提供了依赖注入和AOP等功能,SpringMVC处理HTTP请求和...
这意味着在多线程环境中使用`Hashtable`时无需额外的同步措施,但这也使得其性能较低。 - `HashMap`允许键值为`null`,而`Hashtable`不允许。 - `HashMap`提供了`putIfAbsent`等实用方法,这在`Hashtable`中是没有...
9. **SpringMVC**:Spring框架的Web模块,用于构建MVC应用。 10. **MyBatis**:轻量级持久层框架,结合XML或注解实现SQL映射。 11. **SpringSecurity或Shiro**:安全框架,提供身份验证、授权等功能。 12. **...