问题:java
问题1的简单思考:mysql
成熟的解决方案:spring
使用方式:sql
使用@EnableScheduling注解开启对定时任务的支持。 使用@Scheduled 注解便可,基于corn、fixedRate、fixedDelay等一些定时策略来实现定时任务。数据库
使用缺点:springboot
使用优势:bash
源码分析:服务器
//默认使用的调度器
if(this.taskScheduler == null) {
this.localExecutor = Executors.newSingleThreadScheduledExecutor();
this.taskScheduler = new ConcurrentTaskScheduler(this.localExecutor);
}
//能够看到SingleThreadScheduledExecutor指定的核心线程为1,说白了就是单线程执行
public static ScheduledExecutorService newSingleThreadScheduledExecutor() {
return new DelegatedScheduledExecutorService
(new ScheduledThreadPoolExecutor(1));
}
//利用了DelayedWorkQueue延时队列做为任务的存放队列,这样即可以实现任务延迟执行或者定时执行
public ScheduledThreadPoolExecutor(int corePoolSize) {
super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
new DelayedWorkQueue());
}
复制代码
使用方式:架构
方式一:由1中咱们知道之因此定时任务是阻塞执行,是配置的线程池决定的,那就好办了,换一个不就好了!直接上代码:并发
@Configuration
public class ScheduledConfig implements SchedulingConfigurer {
@Autowired
private TaskScheduler myThreadPoolTaskScheduler;
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
//简单粗暴的方式直接指定
//scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
//也能够自定义的线程池,方便线程的使用与维护,这里很少说了
scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
}
}
@Bean(name = "myThreadPoolTaskScheduler")
public TaskScheduler getMyThreadPoolTaskScheduler() {
ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
taskScheduler.setPoolSize(10);
taskScheduler.setThreadNamePrefix("Haina-Scheduled-");
taskScheduler.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
//调度器shutdown被调用时等待当前被调度的任务完成
taskScheduler.setWaitForTasksToCompleteOnShutdown(true);
//等待时长
taskScheduler.setAwaitTerminationSeconds(60);
return taskScheduler;
}
复制代码
方式二:方式一的本质改变了任务调度器默认使用的线程池,接下来这种是不改变调度器的默认线程池,而是把当前任务交给一个异步线程池去执行
废话太多,直接上代码:
@Scheduled(fixedRate = 1000*10,initialDelay = 1000*20)
@Async("myThreadPoolTaskExecutor")
//@Async
public void scheduledTest02(){
System.out.println(Thread.currentThread().getName()+"--->xxxxx--->"+Thread.currentThread().getId());
}
//自定义线程池
@Bean(name = "myThreadPoolTaskExecutor")
public TaskExecutor getMyThreadPoolTaskExecutor() {
ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
taskExecutor.setCorePoolSize(20);
taskExecutor.setMaxPoolSize(200);
taskExecutor.setQueueCapacity(25);
taskExecutor.setKeepAliveSeconds(200);
taskExecutor.setThreadNamePrefix("Haina-ThreadPool-");
// 线程池对拒绝任务(无线程可用)的处理策略,目前只支持AbortPolicy、CallerRunsPolicy;默认为后者
taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
//调度器shutdown被调用时等待当前被调度的任务完成
taskExecutor.setWaitForTasksToCompleteOnShutdown(true);
//等待时长
taskExecutor.setAwaitTerminationSeconds(60);
taskExecutor.initialize();
return taskExecutor;
}
复制代码
线程池的使用心得
问题:
使用@Scheduled注解来完成设置定时任务,可是有时候咱们每每须要对周期性的时间的设置会作一些改变,或者要动态的启停一个定时任务,那么这个时候使用此注解就不太方便了,缘由在于这个注解中配置的cron表达式必须是常量,那么当咱们修改定时参数的时候,就须要中止服务,从新部署。
解决办法: 方式一:实现SchedulingConfigurer接口,重写configureTasks方法,从新制定Trigger,核心方法就是addTriggerTask(Runnable task, Trigger trigger) ,不过须要注意的是,此种方式修改了配置值后,须要在下一次调度结束后,才会更新调度器,并不会在修改配置值时实时更新,实时更新须要在修改配置值时额外增长相关逻辑处理。
@Configuration
public class ScheduledConfig implements SchedulingConfigurer {
@Autowired
private TaskScheduler myThreadPoolTaskScheduler;
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
//scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
//能够实现动态调整定时任务的执行频率
scheduledTaskRegistrar.addTriggerTask(
//1.添加任务内容(Runnable)
() -> System.out.println("cccccccccccccccc--->" + Thread.currentThread().getId()),
//2.设置执行周期(Trigger)
triggerContext -> {
//2.1 从数据库动态获取执行周期
String cron = "0/2 * * * * ? ";
//2.2 合法性校验.
// if (StringUtils.isEmpty(cron)) {
// // Omitted Code ..
// }
//2.3 返回执行周期(Date)
return new CronTrigger(cron).nextExecutionTime(triggerContext);
}
);
}
}
复制代码
方式二:使用threadPoolTaskScheduler类可实现动态添加删除功能,固然也可实现执行频率的调整
首先,咱们要认识下这个调度类,它实际上是对java中ScheduledThreadPoolExecutor的一个封装改进后的产物,主要改进有如下几点:
顺便说下ThreadPoolTaskExecutor相对于ThreadPoolExecutor的改进点:
扯了这么多,仍是直接上代码:
@Component
public class DynamicTimedTask {
private static final Logger logger = LoggerFactory.getLogger(DynamicTimedTask.class);
//利用建立好的调度类统一管理
//@Autowired
//@Qualifier("myThreadPoolTaskScheduler")
//private ThreadPoolTaskScheduler myThreadPoolTaskScheduler;
//接受任务的返回结果
private ScheduledFuture<?> future;
@Autowired
private ThreadPoolTaskScheduler threadPoolTaskScheduler;
//实例化一个线程池任务调度类,可使用自定义的ThreadPoolTaskScheduler
@Bean
public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
ThreadPoolTaskScheduler executor = new ThreadPoolTaskScheduler();
return new ThreadPoolTaskScheduler();
}
/**
* 启动定时任务
* @return
*/
public boolean startCron() {
boolean flag = false;
//从数据库动态获取执行周期
String cron = "0/2 * * * * ? ";
future = threadPoolTaskScheduler.schedule(new CheckModelFile(),cron);
if (future!=null){
flag = true;
logger.info - 最佳的logger 来源和相关信息。("定时check训练模型文件,任务启动成功!!!");
}else {
logger.info - 最佳的logger 来源和相关信息。("定时check训练模型文件,任务启动失败!!!");
}
return flag;
}
/**
* 中止定时任务
* @return
*/
public boolean stopCron() {
boolean flag = false;
if (future != null) {
boolean cancel = future.cancel(true);
if (cancel){
flag = true;
logger.info - 最佳的logger 来源和相关信息。("定时check训练模型文件,任务中止成功!!!");
}else {
logger.info - 最佳的logger 来源和相关信息。("定时check训练模型文件,任务中止失败!!!");
}
}else {
flag = true;
logger.info - 最佳的logger 来源和相关信息。("定时check训练模型文件,任务已经中止!!!");
}
return flag;
}
class CheckModelFile implements Runnable{
@Override
public void run() {
//编写你本身的业务逻辑
System.out.print("模型文件检查完毕!!!")
}
}
}
复制代码
到此基于springtask下的定时任务的简单使用算是差很少了,其中难免有些错误的地方,或者理解有偏颇的地方欢迎你们提出来!
分享免费学习资料
针对于还会准备免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料) 为何某些人会一直比你优秀,是由于他自己就很优秀还一直在持续努力变得更优秀,而你是否是还在知足于现状心里在窃喜!但愿读到这的您能点个小赞和关注下我,之后还会更新技术干货,谢谢您的支持!
资料领取方式:加入粉丝群963944895
,私信管理员便可免费领取