Springboot 2.0.* 及低版本定时任务@Scheduled多线程配置

1. 首先在springboot启动类上添加 @EnableScheduling 注解。

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableScheduling;

@EnableScheduling
@SpringBootApplication
public class KittyApiApplication {

    public static void main(String[] args) {
        SpringApplication.run(KittyApiApplication.class, args);
    }

}

2. 将任务类注册到spring容器,然后在其任务方法上添加 @Scheduled 注解,@Scheduled有多个属性:

 ① cron:cron表达式,指定任务在特定时间执行;

 ② fixedDelay:表示上一次任务执行完成后多久再次执行,参数类型为long,单位ms;

 ③ fixedDelayString:与fixedDelay含义一样,只是参数类型变为String;

 ④ fixedRate:表示按一定的频率执行任务,参数类型为long,单位ms;

 ⑤ fixedRateString: 与fixedRate的含义一样,只是将参数类型变为String;

 ⑥ initialDelay:表示延迟多久后第一次执行任务,参数类型为long,单位ms;

 ⑦ initialDelayString:与initialDelay的含义一样,只是将参数类型变为String;

 ⑧ zone:时区,默认为当前时区,一般没有用到。

import lombok.extern.slf4j.Slf4j;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;

/**
 * @author 
 * @version 2019/4/15 下午 05:00
 */
@Slf4j
@Service
public class TimedTaskServiceImpl {

    @Scheduled(cron = "*/10 * * * * ?")
    public void task01() {
        log.info("task01: " + System.currentTimeMillis());
    }

    @Scheduled(cron = "*/10 * * * * ?")
    public void task02() {
        log.info("task02: " + System.currentTimeMillis());
    }

    @Scheduled(cron = "*/10 * * * * ?")
    public void task03() {
        log.info("task03: " + System.currentTimeMillis());
    }
}

3. springboot 2.0 使用的是spring framework 5.0,在spring 3.0 中就引入了TaskScheduler接口进行异步执行和任务调度的抽象,spring默认是以单线程执行任务调度,以下展示如何设置多线程在2.1及以前的版本。

(1)springboot 2.1.*以后可以直接通过在properties中通过属性配置:

# 线程池大小 
spring.task.scheduling.pool.size=10 
# 线程名前缀 
spring.task.scheduling.thread-name-prefix=task-pool-

(2)springboot 2.0及以前就需要实现 SchedulingConfigurer 接口,点到@EnableScheduling注解中,可以看到有SchedulingConfigurer接口、SchedulingTaskRegistrar类和ScheduledAnnotationBeanPostProcessor类等。

SchedulingConfigurer接口内容如下,只有一个抽象方法。

@EnableScheduling 注解的注释里也给出了例子,实现这个SchedulingConfigurer接口就可以实现多线程定时任务。值得注意的是最后一段,将线程池交由spring容器管理,指定销毁回调在Bean销毁时调用线程池的shutdown方法,保证spring容器关闭前销毁线程池中的线程,防止线程未终结而驻留。(如果不指定在linux tomcat中可能会使tomcat进程无法通过shutdown命令关闭,导致内存泄露)

实现SchedulingConfigurer接口的configureTasks()。

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

import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;

@Configuration
public class ScheduleConfig implements SchedulingConfigurer {

    @Bean(destroyMethod="shutdownNow")
    public ScheduledExecutorService taskExecutors() {
        return Executors.newScheduledThreadPool(10);
    }

    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        //参数传入一个线程池
        scheduledTaskRegistrar.setScheduler(taskExecutors());
    }

}

在项目部署到tomcat中,如果按照例子中创建线程池需要指定销毁方法为shutdownNow,shutdown会继续执行并且完成所有未执行的任务,shutdownNow 会清除所有未执行的任务并且在运行线程上调用interrupt() 。

ps:博主由于刚开始未设置销毁方法为shutdownNow,在centos上执行tomcat的shutdown命令无法结束tomcat进程,tomcat结束线程在结束spring容器时,无法结束线程池中线程,线程池中线程在spring容器销毁后还未被杀死,从而导致tomcat进程一直驻留,错误信息如下:(web应用启动了一个线程,但未能停止它,这很可能造成内存泄漏)

16-May-2019 23:19:15.018 INFO [main] org.apache.catalina.core.StandardServer.await A valid shutdown command was received via the shutdown port. Stopping the Server instance.
16-May-2019 23:19:15.019 INFO [main] org.apache.coyote.AbstractProtocol.pause Pausing ProtocolHandler ["http-nio-8080"]
16-May-2019 23:19:15.070 INFO [main] org.apache.coyote.AbstractProtocol.pause Pausing ProtocolHandler ["ajp-nio-8009"]
16-May-2019 23:19:15.121 INFO [main] org.apache.catalina.core.StandardService.stopInternal Stopping service [Catalina]
16-May-2019 23:19:15.240 WARNING [localhost-startStop-1] org.apache.catalina.loader.WebappClassLoaderBase.clearReferencesThreads The web application [kittyapi] appears to have started a thread named [pool-2-thread-1] but has failed to stop it. This is very likely to create a memory leak. Stack trace of thread:
 sun.misc.Unsafe.park(Native Method)
 java.util.concurrent.locks.LockSupport.parkNanos(LockSupport.java:215)
 java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.awaitNanos(AbstractQueuedSynchronizer.java:2078)
 java.util.concurrent.ScheduledThreadPoolExecutor$DelayedWorkQueue.take(ScheduledThreadPoolExecutor.java:1093)
 java.util.concurrent.ScheduledThreadPoolExecutor$DelayedWorkQueue.take(ScheduledThreadPoolExecutor.java:809)
 java.util.concurrent.ThreadPoolExecutor.getTask(ThreadPoolExecutor.java:1074)
 java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1134)
 java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
 java.lang.Thread.run(Thread.java:748)

以上是jdk的Executors创建的线程池,也可以使用spring的线程池ThreadPoolTaskScheduler(ScheduledThreadPoolExecutor的包装)来创建线程池,具体如下:

import lombok.extern.slf4j.Slf4j;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;

import java.util.concurrent.Executor;

@Slf4j
@Configuration
public class ScheduleConfig implements SchedulingConfigurer {

    /**
     * 定时任务线程池
     *
     * @return
     */
    @Bean("scheduledThreadPoolExecutor")
    public Executor scheduledThreadPoolExecutor() {
        log.info("start scheduledThreadPoolExecutor");
        ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
        // 配置核心线程数
        scheduler.setPoolSize(10);
        return scheduler;
    }

    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        // 参数传入一个线程池
        scheduledTaskRegistrar.setScheduler(scheduledThreadPoolExecutor());
    }

}

ThreadPoolTaskScheduler类的继承和实现关系如下:

ThreadPoolTaskScheduler是TaskScheduler接口的实现类,它的父类ExecutorConfigurationSupport重写了销毁方法destroy( )和shutdown( ),通过查看源码可以看到,如果不设置等待任务在关闭容器时完成(waitForTasksToCompleteOnShutdown = true),那么就默认调用了ScheduledThreadPoolExecutor(后续解释为什么是这个类)类的shutdownNo方法

ThreadPoolTaskScheduler类是spring对jdk中ScheduledThreadPoolExecutor类的包装,当创建一个ThreadPoolTaskScheduler对象的bean时,它的内部就已经自动创建了一个默认池大小为1的ScheduledThreadPoolExecutor线程池对象,要注意是在spring容器注册bean时才会去初始化内部的线程池。

下面来看ScheduledTaskRegistrar类,可以通过它里面的方法设置TaskScheduler的子类也就是ThreadPoolTaskScheduler类,所以在实现SchedulingConfigurer接口时可以通过调用ThreadPoolTaskScheduler中的set***方法注入外部线程池。

最后看看ScheduledAnnotationBeanPostProcessor这个类,它内部实例化了一个ScheduledTaskRegistrar对象。

接着往下看,在完成注册的方法里,如果有对象有TaskScheduler或者ScheduledExecutorService对象那么直接使用该对象,往下走如果满足条件定时线程池为null,那么就调用BeanFactory先使用class类型去获取容器中的TaskScheduler的子类,如果容器中存在多个TaskScheduler类型的bean,那么使用默认bean名称去获取,默认名称就是taskScheduler。

接着往下走,可以看到,如果刚开始没有设置TaskScheduler线程池,并且容器中也没有注册TaskScheduler线程池对象,那么ScheduledTaskRegistrar会创建一个单线程线程池来执行定时任务。

总结以上,就会就会发现多线程实现只需要直接在spring容器中注册一个TaskScheduler子类也就是ThreadPoolTaskScheduler就可以了(如果存在多个,要指定bean名称为taskScheduler,spring默认使用此名称的bean,这里只是设置了线程数量,其他属性可根据具体需求设置),就不用去实现SchedulingConfigurer接口了。如果定时任务较多复杂,建议集成Quartz

import lombok.extern.slf4j.Slf4j;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;

import java.util.concurrent.Executor;
import java.util.concurrent.ThreadPoolExecutor;

@Slf4j
@Configuration
public class ExecutorConfig {

    @Bean
    public TaskScheduler taskScheduler() {
        ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
        scheduler.setPoolSize(10);
        return scheduler;
    }
}

 

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值