CMS-Spring Task定时任务

2020-01-14 14:45:48 浏览数 (1)

3 Spring Task定时任务

3.1需求分析

根据分布式事务的研究结果,订单服务需要定时扫描任务表向MQ发送任务。本节研究定时任务处理的方案,并实 现定时任务扫描任务表并向MQ发送消息。

实现定时任务的方案如下:

  1. 使用jdk的Timer和TimerTask实现 可以实现简单的间隔执行任务,无法实现按日历去调度执行任务。
  2. 使用Quartz实现 Quartz 是一个异步任务调度框架,功能丰富,可以实现按日历调度。
  3. 使用Spring Task实现 Spring 3.0后提供Spring Task实现任务调度,支持按日历调度,相比Quartz功能稍简单,但是在开发基本够用,支持注解编程方式。 本项目使用Spring Task实现任务调度。

3.2 Spring Task串行任务

3.2.1 编写任务类

在Spring boot启动类上添加注解:@EnableScheduling 新建任务测试类TestTask,编写测试方法如下:

代码语言:javascript复制
@Component
public class ChooseCourseTask {
    
    private static final Logger LOGGER = LoggerFactory.getLogger(ChooseCourseTask.class);

    // @Scheduled(fixedRate = 5000)
    // 上次执行开始时间后5秒执行
    // @Scheduled(fixedDelay = 5000)
    // 上次执行完毕后5秒执行
    // @Scheduled(initialDelay=3000, fixedRate=5000)
    // 第一次延迟3秒,以后每隔5秒执行一次 @Scheduled(cron="0/3 * * * * *")
    // 每隔3秒执行一次
    public void task1() {
        LOGGER.info("===============测试定时任务1开始===============");
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        LOGGER.info("===============测试定时任务1结束===============");
    }
}

测试:

  1. 测试fixedRate和fixedDelay的区别
  2. 测试串行执行的特点

3.2.2 cron表达式

cron表达式包括6部分:

  1. 秒(0~59)
  2. 分钟(0~59)
  3. 小时(0~23)
  4. 月中的天(1~31)
  5. 月(1~12)
  6. 周中的天 (填写MON,TUE,WED,THU,FRI,SAT,SUN,或数字1~7 1表示MON,依次类推)

特殊字符介绍:

  1. “/”字符表示指定数值的增量
  2. “*”字符表示所有可能的值
  3. “-”字符表示区间范围
  4. “,” 字符表示列举
  5. “?”字符仅被用于月中的天和周中的天两个子表达式,表示不指定值

例子:

  1. 0/3 * * * * * 每隔3秒执行
  2. 0 0/5 * * * * 每隔5分钟执行
  3. 0 0 0 * * * 表示每天0点执行
  4. 0 0 12 ? * WEN 每周三12点执行
  5. 0 15 10 ? * MON-FRI 每月的周一到周五10点 15分执行
  6. 0 15 10 ? * MON,FRI 每月的周一和周五10点 15分执行

3.2.3 串行任务测试

参考 task1方法的的定义方法,再定义task2方法,此时共用两个任务方法。

代码语言:javascript复制
    @Scheduled(fixedRate = 3000)
    //上次执行开始时间后5秒执行 
    public void task2() {
        LOGGER.info("===============测试定时任务2开始===============");
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        LOGGER.info("===============测试定时任务2结束===============");
    }

通过测试发现,两个任务方法由一个线程串行执行,task1方法执行完成task2再执行。

3.3 Spring Task并行任务

3.3.1 需求分析

在项目通常是需要多个不同的任务并行去执行。 本节实现Spring Task并行执行任务的方法。

3.3.2 配置异步任务

创建异步任务配置类,需要配置线程池实现多线程调度任务。

代码语言:javascript复制
@Configuration
@EnableScheduling
public class AsyncTaskConfig implements SchedulingConfigurer, AsyncConfigurer {
    //线程池线程数量 
    private int corePoolSize = 5;

    @Bean
    public ThreadPoolTaskScheduler taskScheduler() {
        ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
        scheduler.initialize();
        //初始化线程池 
        scheduler.setPoolSize(corePoolSize);
        //线程池容量 
        return scheduler;
    }

    @Override
    public Executor getAsyncExecutor() {
        Executor executor = taskScheduler();
        return executor;
    }

    @Override
    public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
        return null;
    }

    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        scheduledTaskRegistrar.setTaskScheduler(taskScheduler());
    }
}

@EnableScheduling添加到此配置类上,SpringBoot启动类上不用再添加@EnableScheduling

3.3.3 测试

通过测试发现两个任务由不同的线程在并行执行,互不影响。

0 人点赞