SpringBoot(十三):springboot2.0.2定时任务
使用定义任务:
第一步:启用定时任务
第二步:配置定时器资源等
第三步:定义定时任务并指定触发规则
1)启动类启用定时任务
在springboot入口类上添加注解@EnableScheduling即可。
@SpringBootApplication(scanBasePackages = {})
@MapperScan("com.dx.jobmonitor.mapper")
@EnableScheduling
public class App {
private static final Logger logger = LoggerFactory.getLogger(App.class); public static void main(String[] args) {
logger.info("App start...");
SpringApplication.run(App.class, args);
}
}
2)配置定时任务资源等:
设置定时任务线程池大小:通过SchedulingConfigurer接口配置并行方式
当定时任务很多的时候,为了提高任务执行效率,可以采用并行方式执行定时任务,任务之间互不影响,只要实现SchedulingConfigurer接口就可以。
/**
* 配置定时任务<br>
* 1)当定时任务很多的时候,为了提高任务执行效率,可以采用并行方式执行定时任务,任务之间互不影响,只要实现SchedulingConfigurer接口就可以。<br>
* 2)这里指定用3个线程来并行处理
* **/
@Configuration
public class ScheduledConfig implements SchedulingConfigurer {
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.setScheduler(setTaskExecutors());
} @Bean(destroyMethod = "shutdown")
public Executor setTaskExecutors() {
return Executors.newScheduledThreadPool(3);
}
}
在并行执行的时候,创建线程池采用了newScheduledThreadPool这个线程池。
Executors框架中存在几种线程池的创建线程池的采用的队列是延迟队列:
- newCachedThreadPool() ,
- newFixedThreadPool(),
- newSingleThreadExecutor(),
- newScheduledThreadPool()。
newScheduledThreadPool() 线程池的特性是定时任务能够定时或者周期性的执行任务。
public ScheduledThreadPoolExecutor(int corePoolSize) {
super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS, new DelayedWorkQueue());
}
其中线程池核心线程数是自己设定的,最大线程数是最大值。阻塞队列是自定义的延迟队列:DelayedWorkQueue()
3)定义定时任务
定时任务1:
@Component
public class SchedulerTask {
private int count=0; @Scheduled(cron="*/6 * * * * ?")
private void process(){
System.out.println("this is scheduler task runing "+(count++));
}
}
定时任务2:
从application.yml中读取cron参数
@Component
public class Scheduler2Task {
private static final SimpleDateFormat dateFormat = new SimpleDateFormat("HH:mm:ss"); @Scheduled("jobs.scheduled.cron")
public void reportCurrentTime() {
System.out.println("现在时间:" + dateFormat.format(new Date()));
}
}
此时application.yml中配置信息如下:
jobs:
scheduled:
cron: 0/30 * * * * ?
参数说明:
@Scheduled接受两种定时的设置:
- 一种是cornexpression。
- 一种是Rate/Delay表达式(毫秒值):
- @Scheduled(fixedRate = 6000):上一次开始执行时间点后每隔6秒执行一次。
- @Scheduled(fixedDelay = 6000):上一次执行完毕时间点之后6秒再执行。
- @Scheduled(initialDelay=1000, fixedRate=6000):第一次延迟1秒后执行,之后按fixedRate的规则每6秒执行一次。
动态修改scheduled的cron参数:
动态修改定时任务cron参数时:
- 1)不需要重启应用就可以动态的改变Cron表达式的值
- 2)不能使用@Scheduled(cron = “${jobs.cron}”)实现
@Component
public class SpringDynamicCronTask implements SchedulingConfigurer {
private static final Logger logger = LoggerFactory.getLogger(SpringDynamicCronTask.class);
private static String cron = "0/5 * * * * ?"; @Autowired
private TaskDynamicCronService taskCronService; @Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.addTriggerTask(() -> {
// 任务逻辑
logger.error("dynamicCronTask is running...");
}, triggerContext -> {
// 任务触发,在这里可修改任务的执行周期,因为每次调度都会执行这里
// 1)这里可以修改为从数据读取cron
// cron=taskCronService.getCron();
CronTrigger cronTrigger = new CronTrigger(cron);
return cronTrigger.nextExecutionTime(triggerContext);
});
} /**
* 2) 供应用端调用动态修改cron参数方法
* @Controller
* @RequestMapping("/cron")")
* public class CronController{
* @Autowired
* private SpringDynamicCronTask cronTask;
*
* @PostMapping("/update")
* @ResponseBody
* public String update(String cron) {
* cronTask.setCron(cron);
* }
* }
*/
public void setCron(String cron) {
this.cron=cron;
}
}
动态设置cron参数常用方式包含两种:
1)动态查询并设置cron
定义CronTrigger时,从数据库中动态查询cron并设置
@Autowired
private TaskDynamicCronService taskCronService; @Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.addTriggerTask(() -> {
// 任务逻辑
logger.error("dynamicCronTask is running...");
}, triggerContext -> {
cron=taskCronService.getCron();
CronTrigger cronTrigger = new CronTrigger(cron);
return cronTrigger.nextExecutionTime(triggerContext);
});
}
2)通过接口函数修改cron值
SpringDynamicCronTask 类,提供修改cron函数setCron()。
@Component
public class SpringDynamicCronTask implements SchedulingConfigurer {
private static final Logger logger = LoggerFactory.getLogger(SpringDynamicCronTask.class);
private static String cron = "0/5 * * * * ?";
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.addTriggerTask(() -> {
// 任务逻辑
logger.error("dynamicCronTask is running...");
}, triggerContext -> {
CronTrigger cronTrigger = new CronTrigger(cron);
return cronTrigger.nextExecutionTime(triggerContext);
});
} /**
* 供应用端调用动态修改cron参数方法
*/
public void setCron(String cron) {
this.cron=cron;
}
}
应用端调用:
@Controller
@RequestMapping("/cron")")
public class CronController{
@Autowired
private SpringDynamicCronTask cronTask; @PostMapping("/update")
@ResponseBody
public String update(String cron) {
cronTask.setCron(cron);
}
}
多定时任务管理
定义多定时任务管理类
class BatchTaskSchedule {
private ThreadPoolTaskScheduler executor = new ThreadPoolTaskScheduler();
private ScheduledFuture<?> future;
private Integer key; public BatchTaskSchedule(Integer key) {
this.key = key;
} public void start() {
executor.setPoolSize(1);
executor.setThreadNamePrefix("taskExecutor-");
executor.setWaitForTasksToCompleteOnShutdown(true);
executor.setAwaitTerminationSeconds(60);
// 必须得先初始化,才能使用
executor.initialize();
future = executor.schedule(new Runnable() {
@Override
public void run() {
System.out.println("[" + Thread.currentThread().getName() + "-" + key + "]Hello "
+ new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
}
}, new CronTrigger("0/15 * * * * ?"));
} public void restart() {
// 先停止,在开启.
stop();
start();
} public void stop() {
if (future != null) {
future.cancel(true);
}
}
}
使用测试示例:
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = { App.class }, webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
public class CommonTest {
// 定时任务存储集合:当需要停止、重启任务时,可以从该集合中获取待操作任务。
private static Map<Integer, BatchTaskSchedule> taskBack = new HashMap<Integer, BatchTaskSchedule>(); @Test
public void test() {
for (int i = 0; i < 2; i++) {
// 创建定时任务
BatchTaskSchedule taskScheduled = new BatchTaskSchedule(i);
taskScheduled.start();
taskBack.put(i, taskScheduled);
} try {
Thread.sleep(1 * 60 * 1000);
// 停止掉某个任务
taskBack.get(0).stop();
Thread.sleep(1 * 60 * 1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
SpringBoot(十三):springboot2.0.2定时任务的更多相关文章
- 【SpringBoot】SpringBoot2.0响应式编程
========================15.高级篇幅之SpringBoot2.0响应式编程 ================================ 1.SprinBoot2.x响应 ...
- 【SpringBoot】SpringBoot2.x整合定时任务和异步任务处理
SpringBoot2.x整合定时任务和异步任务处理 一.项目环境 springboot2.x本身已经集成了定时任务模块和异步任务,可以直接使用 二.springboot常用定时任务配置 1.在启动类 ...
- Springboot:SpringBoot2.0整合WebSocket,实现后端数据实时推送!
一.什么是WebSocket? B/S结构的软件项目中有时客户端需要实时的获得服务器消息,但默认HTTP协议只支持请求响应模式,这样做可以简化Web服务器,减少服务器的负担,加快响应速度,因为服务器不 ...
- SpringBoot2.0源码分析(一):SpringBoot简单分析
SpringBoot2.0简单介绍:SpringBoot2.0应用(一):SpringBoot2.0简单介绍 本系列将从源码角度谈谈SpringBoot2.0. 先来看一个简单的例子 @SpringB ...
- SpringBoot2.0 基础案例(04):定时任务和异步任务的使用方式
一.定时任务 1.基本概念 按照指定时间执行的程序. 2.使用场景 数据分析 数据清理 系统服务监控 二.同步和异步 1.基本概念 同步调用 程序按照代码顺序依次执行,每一行程序都必须等待上一行程序执 ...
- springboot学习入门简易版三---springboot2.0启动方式
2.4使用@componentscan方式启动 2.4.1 @EnableAutoConfiguration 默认只扫描当前类 @EnableAutoConfiguration 默认只扫描当前类,如果 ...
- springboot学习入门简易版二---springboot2.0项目创建
2 springboot项目创建(5) 环境要求:jdk1.8+ 项目结构: 2.1创建maven工程 Group id :com.springbootdemo Artifact id: spring ...
- SpringBoot(十一):springboot2.0.2下配置mybatis generator环境,并自定义字段/getter/settetr注释
Mybatis Generator是供开发者在mybatis开发时,快速构建mapper xml,mapper类,model类的一个插件工具.它相对来说对开发者是有很大的帮助的,但是它也有不足之处,比 ...
- 零基础快速入门SpringBoot2.0教程 (三)
一.SpringBoot Starter讲解 简介:介绍什么是SpringBoot Starter和主要作用 1.官网地址:https://docs.spring.io/spring-boot/doc ...
随机推荐
- Ansible 详解
原文:https://www.cnblogs.com/keerya/p/7987886.html#_label0,有改动 一.Ansible简介 1.ansible是什么 a.ansible是新出现的 ...
- 51Nod1317 相似字符串对 容斥原理 动态规划
原文链接https://www.cnblogs.com/zhouzhendong/p/51Nod1317.html 题目传送门 - 51Nod1317 题意 称一对字符串(A,B)是相似的,当且仅当满 ...
- 058 kafka与log4j集成
1.首先在resources下面写log4j.properties 主要是因为kafka.producer.KafkaLog4jAppender类的存在. log4j.rootLogger=INFO, ...
- Trident继承kafka
1.Kafka涉及的类 上一个类是不透明事务 后一个是完全事务 2.启动服务 3..驱动类 重要的地方是修改了两个部分: 1.数据的来源是kafka 2.第二个是字段的Fields是str packa ...
- 关于thinkphp3自动完成的笔记
当我在前台传入的主键id与字段表的主键id值时,在更新时tp总是判断为新增的状态(解决办法:将前台的表单主键名保持和数据表主键id名一只,手动创建数据) create时是先获取主键id判断'$type ...
- 转载:ThreadPoolExecutor 源码阅读
前言 之前研究了一下如何使用ScheduledThreadPoolExecutor动态创建定时任务(Springboot定时任务原理及如何动态创建定时任务),简单了解了ScheduledThreadP ...
- Pandas 学习记录(一)
1.DataFrame 按照列和按照行进行索引数据 按照列索引 df[’column_name’] 按照行索引 df.loc[’row_key’] 或 df.iloc[index] 2.先行后列索引单 ...
- mysql查询根据时间排序
表数据: mysql查询根据时间排序,如果有相同时间则只查询出来一个 所以需要再判断,如果时间相同,则根据id进行降序排序
- 2186 ACM 水题 int 向下取整
题目:http://acm.hdu.edu.cn/showproblem.php?pid=2186 扩展: #include <cstdio> 使用floor函数.floor(x)返回的是 ...
- Scrapy基础(二)————Scrapy的安装和目录结构
Scrapy安装: 1,首先进入虚拟环境 2,使用国内豆瓣源进行安装,快! pip install -i https://pypi.douban.com/simple/ scrapy 3,特殊情 ...