重庆分公司,新征程启航

为企业提供网站建设、域名注册、服务器等服务

SpringBoot多线程执行task任务

SpringBoot添加定时任务入门请看这里=>SpringBoot入门八,添加定时任务

创新互联建站是一家专业提供白塔企业网站建设,专注与网站设计制作、做网站H5页面制作、小程序制作等业务。10年已为白塔众多企业、政府机构等服务。创新互联专业的建站公司优惠进行中。

一.问题描述

Task定时任务默认都是使用单线程执行的,如果定时任务有很多的话,那么可能会导致很多任务无法按时准确执行,示例如下:

import java.text.SimpleDateFormat;
import java.util.Date;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

@Component
public class TaskTest {
    private final Logger log = LoggerFactory.getLogger(TaskTest02.class);
    //输出时间格式
    private static final SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss:sss");

    @Scheduled(cron = "0/15 * * * * ? ")
    private void sayHello(){
        String dateTime = format.format(new Date());
        log.info("{} 向宇宙发出了一声问候: Hello World!", dateTime);
        try {
            Thread.sleep(10000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    @Scheduled(cron = "0/16 * * * * ? ")
    private void sayHello2(){
        String dateTime = format.format(new Date());
        log.info("{} 向宇宙发出了一声问候: 你好,世界", dateTime);
    }
}

当sayHello()方法执行的时候,因为长时间占用任务执行线程,导致sayHello2()被迫向后延时执行,如图:

SpringBoot多线程执行task任务

二.解决方案
方案1

添加以下代码块,可放置在任意一个类中,整个工程只需要添加一个即可

@Bean
public TaskScheduler taskScheduler() {
    ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
    // 设置scheduler执行线程为3个
    scheduler.setPoolSize(3);
    return scheduler;
}

SpringBoot多线程执行task任务

方案2(个人推荐)

添加一个配置类即可,定时任务类或方法不用做任何改变

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;

/**
 * @描述: 多线程执行定时任务
 * @日期 2019年5月28日
 */
@Configuration
public class TaskConfig {
    /**
     * @描述: 所有的定时任务都放在一个线程池中,定时任务启动时使用不同的线程
     * @return
     * @日期 2019年5月28日
     */
    @Bean
    public TaskScheduler taskScheduler() {
        ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
        // 设置scheduler执行线程为3个
        scheduler.setPoolSize(3);
        return scheduler;
    }
}
方案3

添加一个配置类即可(实现SchedulingConfigurer接口),定时任务类或方法不用做任何改变

import java.util.concurrent.Executors;

import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;

/**
 * @描述: 多线程执行定时任务
 * @日期 2019年5月27日
 */
@Configuration
public class SchedulingConfig implements SchedulingConfigurer {
    /**
     * @描述: 所有的定时任务都放在一个线程池中,定时任务启动时使用不同的线程
     * @param taskRegistrar
     * @日期 2019年5月27日
     */
    @Override
    public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
        //设定一个定时任务线程池,数量为3
        taskRegistrar.setScheduler(Executors.newScheduledThreadPool(3));
    }
}
三.执行结果

可以看到两个定时任务已经分别由不同的线程执行了

SpringBoot多线程执行task任务


文章名称:SpringBoot多线程执行task任务
文章位置:http://cqcxhl.com/article/poisgc.html

其他资讯

在线咨询
服务热线
服务热线:028-86922220
TOP