IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> @Schedule定时任务是并行执行吗 -> 正文阅读

[Java知识库]@Schedule定时任务是并行执行吗

@Schedule是常用的定时任务注解,一般用在需要定时执行的业务逻辑上,多用于单机任务,分布式任务使用的话,需要通过分布式锁保证数据一致性。

如果有多个@Schedule注解定义的定时任务,是并发执行还是串行执行呢?

@Schedule定义的定时任务应该也是有一个线程池,例如4大常见线程池中的ScheduleThreadPool

四大常见线程池:

  1. SingleThreadPool:单线程的线程池
  2. FixedThreadPool:固定线程的线程池
  3. ScheduleThreadPool:定时执行的任务的线程池
  4. CachedThreadPool:缓存线程数但是线程数量无限大的线程池

那么是并发执行多个定时任务还是只有一个线程串行执行定时任务呢?

如果是串行执行,那么会有严重的问题!定时任务不能按时执行,并且会有阻塞的风险

那么如何让@Schedule定义的定时任务并发多线程执行呢?

测试

在同一个类中,通过@Schedule定义多个定时任务,查看多个定时任务是否使用同一个线程执行,并且是否为并行执行

  • 示例代码
@Component
public class ScheduleTest {

    @Scheduled(cron = "0/30 * * * * ?")
    public void task1() {
        System.out.println("task1 start");
        System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
    }


    @Scheduled(cron = "0/30 * * * * ?")
    public void task2() {
        System.out.println("task2 start");
        System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
    }

    @Scheduled(cron = "0/50 * * * * ?")
    public void task3() {
        System.out.println("task3 start");
        System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
    }

}
  • 输出
task2 start
40  scheduling-1
task1 start
40  scheduling-1
task3 start
40  scheduling-1

可以看到,三个任务使用的是同一个线程

  • 再次测试

通过在某一个任务中写一个死循环,让这个任务一直再执行,然后看看其他的定时任务是否还会执行

   @Scheduled(cron = "0/30 * * * * ?")
    public void task1() {
        System.out.println("task1 start");
        System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
    }


    @Scheduled(cron = "0/30 * * * * ?")
    public void task2() {
        while (true) {
            System.out.println("task2 start");
            System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
        }
    }

通过输出可以看到,task1的任务一直没有执行,task2的任务一直在执行

源码分析

@Schedule 注解的处理类在org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor这个类中

  1. 在这个方法中,表示该bean初始化完成之后,执行的业务逻辑

image-20220503220620941

  1. finishRegistration方法最下面的this.registrar.afterPropertiesSet();方法中,表示开始调用后置处理器。

  2. 后置处理器中则开始执行上面扫描到的所有的定时任务,org.springframework.scheduling.config.ScheduledTaskRegistrar#scheduleTasks

  3. 通过scheduleTasks方法中可以看到,在执行时,会检查是否有自定义线程池,如果没有,那么会创建一个SingleThreadSchedulePool 作为线程池执行

    image-20220503221121497

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vLHH8D3n-1651588669675)(https://cdn.jsdelivr.net/gh/chenliang15405/picture/hub/cs/image-20220503221258957.png)]

看到这里,已经很明显了,在使用的时候,没有自定义线程池,所以导致在执行任务的时候,Spring会自动创建一个线程池去执行,但是这个默认的线程池是一个核心线程为1的单线程的线程池

解决方案

  • 第一种:全局配置

    这种方式相当于切面的方式,对统一的定时任务对处理,无需关注每个定时任务的线程池配置

    @Configuration
    @EnableScheduling
    public class ScheduleConfig implements SchedulingConfigurer {
    
    
        @Override
        public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
            taskRegistrar.setScheduler(Executors.newScheduledThreadPool(Runtime.getRuntime().availableProcessors()));
        }
    
    }
    
  • 第二种:自定义线程池

    这种方式可以根据业务逻辑以及定时任务的重要性,配置不同的线程池,对不同的任务对隔离,互不影响,这种方式配置的定时任务更灵活

    • 配置类

      @EnableAsync
      @Configuration
      public class AsyncScheduleConfig {
      
          @Bean("scheduleExecutor")
          public Executor myAsync() {
              ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
              //最大线程数
              executor.setMaxPoolSize(100);
              //核心线程数
              executor.setCorePoolSize(10);
              //任务队列的大小
              executor.setQueueCapacity(10);
              //线程前缀名
              executor.setThreadNamePrefix("async-schedule-");
              //线程存活时间
              executor.setKeepAliveSeconds(60);
              /**
               * 拒绝处理策略
               * CallerRunsPolicy():交由调用方线程运行,比如 main 线程。
               * AbortPolicy():直接抛出异常。
               * DiscardPolicy():直接丢弃。
               * DiscardOldestPolicy():丢弃队列中最老的任务。
               */
              executor.setRejectedExecutionHandler(new ThreadPoolExecutor.AbortPolicy());
              //线程初始化
              executor.initialize();
              return executor;
          }
      
      }
      
    • 使用

      只需要在定时任务的注解上面加上@Async注解,并显式指明自定义的线程池名称即可

      @Component
      public class ScheduleTest {
      
          @Async("scheduleExecutor")
          @Scheduled(cron = "0/30 * * * * ?")
          public void task1() {
              System.out.println("task1 start");
              System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
          }
      
          @Async("scheduleExecutor")
          @Scheduled(cron = "0/30 * * * * ?")
          public void task2() {
              System.out.println("task2 start");
              System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
          }
      
          @Async("scheduleExecutor")
          @Scheduled(cron = "0/50 * * * * ?")
          public void task3() {
              System.out.println("task3 start");
              System.out.println(Thread.currentThread().getId() + "  " + Thread.currentThread().getName());
          }
      
      }
      
  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-05-05 11:04:09  更:2022-05-05 11:07:55 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 0:16:51-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码