timer
1.需要关注两个类,Timer和TimerTask
TimerTask:任务内容
Timer:里面有调度TimerTask执行的方法
2.适用于比较简单的计时任务(简单任务下或许比Quartz用起来更方便,不用引入额外的包,逻辑也没有那么复杂)
3.两个类均位于java.util包下
4.如果出现错误,程序会直接崩溃,而不会捕获异常进行异常处理
5.代码
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("定时任务测试");
}
}
Quartz
1.Quartz是什么
Quartz是一个完全由java编写的开源作业调度框架。不要让作业调度这个术语吓着你。尽管Quartz框架整合了许多额外功能, 但就其简易形式看,你会发现它易用得简直让人受不了!
————根据我的使用经验,确实是相当的简单,而且逻辑思路也非常的清晰
2.Quartz能做什么
假设你有以下需求,或许可以使用它:
–想要每天凌晨1点统计昨天新增的用户
–想要每隔一小时备份自己数据库的数据
–想要在特定的节日提醒自己做某些事情
–想要代替人工,每半个小时自动处理邮件
–定期清理数据库不需要的数据
在需要做重复功的时候就可以考虑使用任务调度机制了
3.如何使用
(1)需要使用外部jar包,先引入相关包,我用的版本是2.3.2
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz</artifactId>
<version>${quartz.version}</version>
</dependency>
(2)定义一个类,实现Job接口。
需要重写Job接口的execute。这里需要注意的是JobExecutionContext这个参数,通过它可以获得所有运行相关的参数或者是专门传过来的值,在方法内,主要写需要执行的逻辑,至于怎么执行由其它地方规定。
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);
}
}
Quzrtz与SpringBoot整合
1.导入依赖包
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-quartz</artifactId>
</dependency>
2.编写任务
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() + ": 任务正在执行!");
}
}
3.编写触发器和调度器
实现了CommandLineRunner接口之后,在程序启动的时候就会触发run函数内的任务执行了。整合好后可以开始启动程序了
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整合
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和测试任务注解的值时对应的。通过查看日志即可知道是否执行成功。