原创不易,如需转载,请注明出处http://www.javashuo.com/article/p-zqnnuvrr-cx.html,不然将追究法律责任!!! html
Timer:jdk中自带的一个定时调度类,能够简单的实现按某一频度进行任务执行。提供的功能比较单一,没法实现复杂的调度任务。
ScheduledExecutorService:也是jdk自带的一个基于线程池设计的定时任务类。其每一个调度任务都会分配到线程池中的一个线程执行,因此其任务是并发执行的,互不影响。
Spring Task:Spring提供的一个任务调度工具,支持注解和配置文件形式,支持Cron表达式,使用简单但功能强大。
Quartz:一款功能强大的任务调度器,能够实现较为复杂的调度功能,如每个月一号执行、天天凌晨执行、每周五执行等等,还支持分布式调度,就是配置稍显复杂。
I、如何解决定时任务的屡次执行? II、如何解决任务的单点问题,实现任务的故障转移?
一、固定执行定时任务的机器(能够有效避免屡次执行的状况 ,缺点就是单点故障问题)。 二、借助Redis的过时机制和分布式锁。 三、借助mysql的锁机制等。
一、Quartz:能够去看看这篇文章[Quartz分布式]( https://www.cnblogs.com/jiafuwei/p/6145280.html)。 二、elastic-job:(https://github.com/elasticjob/elastic-job-lite)当当开发的弹性分布式任务调度系统,采用zookeeper实现分布式协调,实现任务高可用以及分片。 三、xxl-job:(https://github.com/xuxueli/xxl-job)是大众点评员发布的分布式任务调度平台,是一个轻量级分布式任务调度框架。 四、saturn:(https://github.com/vipshop/Saturn) 是惟品会提供一个分布式、容错和高可用的做业调度服务框架。
使用@EnableScheduling注解开启对定时任务的支持。 使用@Scheduled 注解便可,基于corn、fixedRate、fixedDelay等一些定时策略来实现定时任务。
一、多个定时任务使用的是同一个调度线程,因此任务是阻塞执行的,执行效率不高。 二、其次若是出现任务阻塞,致使一些场景的定时计算没有实际意义,好比天天12点的一个计算任务被阻塞到1点去执行,会致使结果并不是咱们想要的。
一、配置简单 二、适用于单个后台线程执行周期任务,而且保证顺序一致执行的场景
//默认使用的调度器 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中咱们知道之因此定时任务是阻塞执行,是配置的线程池决定的,那就好办了,换一个不就好了!直接上代码:java
@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; }
方式二:方式一的本质改变了任务调度器默认使用的线程池,接下来这种是不改变调度器的默认线程池,而是把当前任务交给一个异步线程池去执行mysql
废话太多,直接上代码:git
@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; }
线程池的使用心得(后续有专门文章来探讨)github
使用@Scheduled注解来完成设置定时任务,可是有时候咱们每每须要对周期性的时间的设置会作一些改变,或者要动态的启停一个定时任务,那么这个时候使用此注解就不太方便了,缘由在于这个注解中配置的cron表达式必须是常量,那么当咱们修改定时参数的时候,就须要中止服务,从新部署。spring
方式一:实现SchedulingConfigurer接口,重写configureTasks方法,从新制定Trigger,核心方法就是addTriggerTask(Runnable task, Trigger trigger) ,不过须要注意的是,此种方式修改了配置值后,须要在下一次调度结束后,才会更新调度器,并不会在修改配置值时实时更新,实时更新须要在修改配置值时额外增长相关逻辑处理。sql
@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的一个封装改进后的产物,主要改进有如下几点: 一、提供默认配置,由于是ScheduledThreadPoolExecutor,因此只有poolSize这一个默认参数。 二、支持自定义任务,经过传入Trigger参数。 三、对任务出错处理进行优化,若是是重复性的任务,不抛出异常,经过日志记录下来,不影响下次运行,若是是只执行一次的任务,将异常往上抛。 顺便说下ThreadPoolTaskExecutor相对于ThreadPoolExecutor的改进点: 一、提供默认配置,原生的ThreadPoolExecutor的除了ThreadFactory和RejectedExecutionHandler其余没有默认配置 二、实现AsyncListenableTaskExecutor接口,支持对FutureTask添加success和fail的回调,任务成功或失败的时候回执行对应回调方法。 三、由于是spring的工具类,因此抛出的RejectedExecutionException也会被转换为spring框架的TaskRejectedException异常(这个无所谓) 四、提供默认ThreadFactory实现,直接经过参数重载配置
扯了这么多,仍是直接上代码:segmentfault
@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("定时check训练模型文件,任务启动成功!!!"); }else { logger.info("定时check训练模型文件,任务启动失败!!!"); } return flag; } /** * 中止定时任务 * @return */ public boolean stopCron() { boolean flag = false; if (future != null) { boolean cancel = future.cancel(true); if (cancel){ flag = true; logger.info("定时check训练模型文件,任务中止成功!!!"); }else { logger.info("定时check训练模型文件,任务中止失败!!!"); } }else { flag = true; logger.info("定时check训练模型文件,任务已经中止!!!"); } return flag; } class CheckModelFile implements Runnable{ @Override public void run() { //编写你本身的业务逻辑 System.out.print("模型文件检查完毕!!!") } } }
我的博客地址:springboot
csdn:https://blog.csdn.net/tiantuo6513
cnblogs:https://www.cnblogs.com/baixianlong
segmentfault:https://segmentfault.com/u/baixianlong github:https://github.com/xianlongbai