一.Timer
package com.shixin.pawcode.resources.timer;
/**
* @Description
* @Author shixin
* @Date 2021/5/12 8:51
*/
public class MyTimer {
private static int count = 0;
public static void main(String[] args) {
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
//延迟2s开始执行,执行间隔1s
System.out.println("当前时间 : "+sdf.format(new Date()));
new Timer().schedule(new TimerTask() {
@Override
public void run() {
System.out.println(count++ + " : " +sdf.format(new Date()));
}
}, 2000L, 1000L);
}
}
执行结果
二.@Scheduled
这个注解是Spring自带的,直接按照步骤使用即可。
1.在启动类上添加注解@EnableScheduling
2.代码
package com.shixin.pawcode.resources.config;
@Slf4j
@Component
public class SpringAnnotationScheduler {
@Scheduled(cron = "* * * * * ?")
private void test(){
log.info("定时任务测试");
}
}
这样就行了......不懂参数cron的可以看看这篇文章 :cron详解,在文章下半部分的代码部分也有介绍cron,可以继续往下看。另外附上生成器网址:Cron表达式生成
三.Quartz
1.Quartz是什么
2.Quartz能做什么
3.如何使用
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz</artifactId>
<version>${quartz.version}</version>
</dependency>
(2)定义一个类,实现Job接口。
package com.shixin.pawcode.resources.quartz;
/**
* @Description
* @Author shixin
* @Date 2021/5/12 13:35
*/
public class TestJob implements Job {
private static final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
@Getter
@Setter
private String str;
@Override
public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {
//编写具体的业务逻辑
System.out.println("当前时间:"+sdf.format(new Date()));
//获取JobDataMap 方法一
JobDataMap jobDetailMap = jobExecutionContext.getJobDetail().getJobDataMap();
for (String key : jobDetailMap.keySet()){
System.out.println(key + " : " + jobDetailMap.get(key));
}
//获取JobDataMap 方法二
System.out.println(str);
}
}
(3)三种核心要素
package com.shixin.pawcode.resources.quartz;
/**
* @Description
* @Author shixin
* @Date 2021/5/12 13:38
*/
public class TestScheduler {
private static final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
public static void main(String[] args) throws SchedulerException {
//任务:创建一个JobDetail实例,将该实例与Job绑定 : 任务的描述
JobDataMap detailMap = new JobDataMap();
detailMap.put("str-Map","map-JobDetail");
detailMap.put("int-Map",1);
JobDetail jobDetail = JobBuilder.newJob(TestJob.class)
.withIdentity("jobDetailName","jobDetailGroup")
.usingJobData("str","JobDetail")
.usingJobData("int",-1)
.usingJobData(detailMap)
.build();
//触发器:创建一个Trigger实例 :定义如何执行,什么时候触发,什么时候结束
Calendar startCalendar = Calendar.getInstance();
startCalendar.add(Calendar.SECOND,3);
Calendar endCalendar = Calendar.getInstance();
endCalendar.add(Calendar.SECOND,7);
/*SimpleTrigger trigger = (SimpleTrigger)TriggerBuilder.newTrigger()
.withSchedule(SimpleScheduleBuilder.simpleSchedule()
//执行时间间隔
.withIntervalInSeconds(2)
//永远重复执行 ,这个和执行次数选一个
//.repeatForever()
//执行次数
.withRepeatCount(2))
.withIdentity("triggerName","triggerGroup")
//被调度后马上执行,和具体执行时间选一个
//.startNow()
//被调度后执行的具体的时间
.startAt(startCalendar.getTime())
//被调度后执行的结束的时间,会覆盖重复次数的效果
.endAt(endCalendar.getTime())
.build();*/
/*
* 基于日历的作业调度器,更加强大。重点是Cron表达式掌握
* Cron表达式由7个子表达式组成的字符串,描述了时间的详细信息
[秒] [分] [时] [日] [月] [周] [年]
, 或者
- 之间
* 每
/ 每 三个/人 1分钟/小时
? 不关心
* 举例:
* 1.每天的10:15分触发 [ 0 15 10 ? * * ]
* 2.每天下午的2:00 - 2:55开始(每隔五分钟触发一次) [ 0 0/5 14 * * ? ]
* 3.从周一到周五每天上午的10:15触发 [ 0 15 10 ? * MON-FRI ]
* 4.每月的第三周的星期五的10:15开始触发 [ 0 15 10 ? * 6#3 ]
* 5.从2020-2021年每月最后一周的星期五的10点15分触发 [ 0 15 10 ? * 6L 2020-2021 ]
* */
CronTrigger trigger = (CronTrigger) TriggerBuilder.newTrigger()
.withSchedule(CronScheduleBuilder.cronSchedule("* * * * * ? *"))
.withIdentity("triggerName","triggerGroup")
.startNow()
.build();
//调度器:创建Scheduler实例,并执行,将jobDetail和trigger绑定。
SchedulerFactory schedulerFactory = new StdSchedulerFactory();
Scheduler scheduler = schedulerFactory.getScheduler();
scheduler.start();
System.out.println(sdf.format(new Date()));
scheduler.scheduleJob(jobDetail,trigger);
}
}
四.Quzrtz与SpringBoot整合
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-quartz</artifactId>
</dependency>
package com.shixin.pawcode.resources.quartz.job;
public class Job1 extends QuartzJobBean{
@Override
protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
System.out.println(context.getJobDetail().getKey().getName() + ": 任务正在执行!");
}
}
package com.shixin.pawcode.resources.quartz.job;
public class Job2 extends QuartzJobBean{
@Override
protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
System.out.println(context.getJobDetail().getKey().getName() + ": 任务正在执行!");
}
}
3.编写触发器和调度器
package com.shixin.pawcode.resources.config;
import com.shixin.pawcode.resources.quartz.job.Job1;
import org.quartz.*;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.scheduling.quartz.SchedulerFactoryBean;
import org.springframework.stereotype.Component;
/**
* @Description 定时任务配置
* @Author shixin
* @Date 2021/5/13 15:28
*/
@Component
public class CronSchedulerJob implements CommandLineRunner{
@Autowired
private SchedulerFactoryBean schedulerFactoryBean;
@Override
public void run(String... args) throws Exception {
Scheduler scheduler = schedulerFactoryBean.getScheduler();
scheduler.start();
job1(scheduler);
job2(scheduler);
}
private void job1(Scheduler scheduler) throws SchedulerException{
JobDetail jobDetail = JobBuilder.newJob(Job1.class) .withIdentity("job1", "group1").build();
CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule("0/2 * * * * ?");
CronTrigger cronTrigger = TriggerBuilder.newTrigger()
.withIdentity("trigger1", "group1")
.usingJobData("name","shixin")
.withSchedule(scheduleBuilder).build();
scheduler.scheduleJob(jobDetail, cronTrigger);
}
private void job2(Scheduler scheduler) throws SchedulerException{
JobDetail jobDetail = JobBuilder.newJob(Job1.class) .withIdentity("job2", "group2").build();
CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule("0/3 * * * * ?");
CronTrigger cronTrigger = TriggerBuilder.newTrigger()
.withIdentity("trigger2", "group2")
.usingJobData("name","shixin")
.withSchedule(scheduleBuilder).build();
scheduler.scheduleJob(jobDetail, cronTrigger);
}
}
五.xxl-job调度中心的使用以及与SpringBoot整合
[中文文档] / [English Documentation] / [源码仓库地址]
1.一些使用心得
这个分布式调度平台的使用有些类似于注册中心的使用,在注册中心中,应用可以自动通过心跳连接注册中心,注册中心对应用有一定的控制权限。在xxl-job中,有两个重要的"组件",调度中心与执行器,他们的关系就好比注册中心与应用一样。
2.通过源码仓库地址下载源码,如果有时间,建议将doc里面的XXL-JOB官方文档或者阅读上面的[文档],里面前几个步骤可以走完一个完整的流程。
3..按照官方文档的介绍或者下面的配置修改xxl-job-admin的application.properties,然后打包
### 调度中心JDBC链接:链接地址请保持和 2.1章节 所创建的调度数据库的地址一致
spring.datasource.url=jdbc:mysql://127.0.0.1:3306/xxl_job?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghai
spring.datasource.username=root
spring.datasource.password=root_pwd
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
### 报警邮箱
spring.mail.host=smtp.qq.com
spring.mail.port=25
spring.mail.username=xxx@qq.com
spring.mail.password=xxx
spring.mail.properties.mail.smtp.auth=true
spring.mail.properties.mail.smtp.starttls.enable=true
spring.mail.properties.mail.smtp.starttls.required=true
spring.mail.properties.mail.smtp.socketFactory.class=javax.net.ssl.SSLSocketFactory
### 调度中心通讯TOKEN [选填]:非空时启用;
xxl.job.accessToken=
### 调度中心国际化配置 [必填]: 默认为 "zh_CN"/中文简体, 可选范围为 "zh_CN"/中文简体, "zh_TC"/中文繁体 and "en"/英文;
xxl.job.i18n=zh_CN
## 调度线程池最大线程配置【必填】
xxl.job.triggerpool.fast.max=200
xxl.job.triggerpool.slow.max=100
### 调度中心日志表数据保存天数 [必填]:过期日志自动清理;限制大于等于7时生效,否则, 如-1,关闭自动清理功能;
xxl.job.logretentiondays=30
4.根据上面配置的数据库,去里面执行doc/db/tables_xxl_job.sql文件
5.运行打包的文件,访问:http://localhost:8080/xxl-job-admin
6.这个时候就已经配置好调度中心了,但是因为还没有配置执行器,所以无法成功执行任务
7.在你要使用定时任务的项目中引入以下jar包以及添加并且修改配置
<dependency>
<groupId>com.xuxueli</groupId>
<artifactId>xxl-job-core</artifactId>
<version>${最新稳定版本}</version>
</dependency>
### 调度中心部署跟地址 [选填]:如调度中心集群部署存在多个地址则用逗号分隔。执行器将会使用该地址进行"执行器心跳注册"和"任务结果回调";为空则关闭自动注册;
###这里需要填写调度器的地址
xxl.job.admin.addresses=http://127.0.0.1:8080/xxl-job-admin
### 执行器通讯TOKEN [选填]:非空时启用;
xxl.job.accessToken=
### 执行器AppName [选填]:执行器心跳注册分组依据;为空则关闭自动注册
xxl.job.executor.appname=xxl-job-executor-sample
### 执行器注册 [选填]:优先使用该配置作为注册地址,为空时使用内嵌服务 ”IP:PORT“ 作为注册地址。从而更灵活的支持容器类型执行器动态IP和动态映射端口问题。
xxl.job.executor.address=
### 执行器IP [选填]:默认为空表示自动获取IP,多网卡时可手动设置指定IP,该IP不会绑定Host仅作为通讯实用;地址信息用于 "执行器注册" 和 "调度中心请求并触发任务";
xxl.job.executor.ip=
### 执行器端口号 [选填]:小于等于0则自动获取;默认端口为9999,单机部署多个执行器时,注意要配置不同执行器端口;
xxl.job.executor.port=9999
### 执行器运行日志文件存储磁盘路径 [选填] :需要对该路径拥有读写权限;为空则使用默认路径;
xxl.job.executor.logpath=/data/applogs/xxl-job/jobhandler
### 执行器日志文件保存天数 [选填] : 过期日志自动清理, 限制值大于等于3时生效; 否则, 如-1, 关闭自动清理功能;
xxl.job.executor.logretentiondays=30
8.添加配置文件XxlJobConfig.java
9.添加测试任务AdminTask.java 这里只写了一个Bean模式的实例,需要使用其他可以查阅文档
10.如果以上步骤配置正确,在执行器管理中可以看到自己应用内配置的xxl:job:appname的值的执行器。打开任务管理,点击新增,按照图中设置好任务模板,有一点需要注意的是,JobHandler和测试任务注解的值时对应的。通过查看日志即可知道是否执行成功。
后续有时间还会更新记录