SpringBoot并发执行定时任务

本文介绍如何在SpringBoot中实现多线程并发执行定时任务,解决所有任务在同一线程中运行的问题。通过配置ScheduleConfig类实现SchedulingConfigurer接口,使用Executors创建ScheduledThreadPool,使定时任务能在不同线程中并发执行。

运行方法
Spring Boot工程已经集成了服务器。右键点击DemoApplication.java -> Run As -> Java Application即可运行工程。默认端口为8080,启动后在浏览器地址栏输入http://localhost:8080/JobManager.html就可以看到效果。

SpringBoot并发执行定时任务
开启缓存注解
@SpringBootApplication

@EnableScheduling //开启定时任务

public class Application {

       public static void main(String[] args) {

                 SpringApplication.run(Application.class, args);

       }

}
存在问题
发现个问题,通过同时测试几个任务发现,所有的任务都是在同一个线程池中的同一个线程来完成的。在实际开发过程中,我们当然不希望所有的任务都运行在一个线程中。

@Scheduled(cron="0/1 * * * * ? ") //每1秒执行一次

public void testCron1() {

      DateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

      logger.info(sdf.format(new Date())+"*********每1秒执行一次");

}

@Scheduled(cron="0/2 * * * * ? ") //每2秒执行一次

public void testCron2() {

      DateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

      logger.info(sdf.format(new Date())+"*********每2秒执行一次");

}

@Scheduled(cron="0/3 * * * * ? ") //每3秒执行一次

public void testCron3() {

      DateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

      logger.info(sdf.format(new Date())+"*********每3秒执行一次");

}

解决方案
那么,怎么设计成多线程实现并发呢?在网上看到过这样的解决方案。通过ScheduleConfig配置文件实现SchedulingConfigurer接口,并重写setSchedulerfang方法,我们尝试着配置了一下。

@Configuration

public class ScheduleConfig implements SchedulingConfigurer {

      @Override

      public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {

                  taskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));

      }

}

整合成功
这样就完成了多线程并发的配置?我们启动项目通过控制台输出信息验证一下结果,最后发现所有的任务都在同一个线程池但不同线程中完成,说明这个方案完全可行,这样,我们就完成了spring boot 多线程并发定时任务。


@Scheduled所支持的参数:

1.cron:cron表达式,指定任务在特定时间执行;
2.fixedDelay:表示上一次任务执行完成后多久再次执行,参数类型为long,单位ms;
3.fixedDelayString:与fixedDelay含义一样,只是参数类型变为String;
4.fixedRate:表示按一定的频率执行任务,参数类型为long,单位ms;
5.fixedRateString: 与fixedRate的含义一样,只是将参数类型变为String;
6.initialDelay:表示延迟多久再第一次执行任务,参数类型为long,单位ms;
7.initialDelayString:与initialDelay的含义一样,只是将参数类型变为String;
8.zone:时区,默认为当前时区,一般没有用到。

Cron表达式范例:

每隔5秒执行一次:*/5 * * * * ?
每隔1分钟执行一次:0 */1 * * * ?
每天23点执行一次:0 0 23 * * ?
每天凌晨1点执行一次:0 0 1 * * ?
每月1号凌晨1点执行一次:0 0 1 1 * ?
每月最后一天23点执行一次:0 0 23 L * ?
每周星期天凌晨1点实行一次:0 0 1 ? * L
在26分、29分、33分执行一次:0 26,29,33 * * * ?
每天的0点、13点、18点、21点都执行一次:0 0 0,13,18,21 * * ?

其实不会Cron表达式也不用担心,网上有好多在线Cron生成器,我们完全可以通过在线生成器生成符合要求的cron,也很方便。

实例:
import java.util.Date;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.Scheduled;
import com.isoftstone.common.utils.date.DateUtil;

/**

  • @author leon

  • @createDate 2018年5月9日 下午3:25:26

  • @version v1.0

  • @classRemarks 定时器测试
    */
    @Configuration
    @EnableScheduling
    public class TestTask {

    private static Logger logger = LoggerFactory.getLogger(TestTask.class);

    //每隔3秒钟审批一次 (cron 表达式)
    //Cron表达式范例:
    //每隔5秒执行一次:*/5 * * * * ?
    //每隔1分钟执行一次:0 */1 * * * ?
    //每天23点执行一次:0 0 23 * * ?
    //每天凌晨1点执行一次:0 0 1 * * ?
    //每月1号凌晨1点执行一次:0 0 1 1 * ?
    //每月最后一天23点执行一次:0 0 23 L * ?
    //每周星期天凌晨1点实行一次:0 0 1 ? * L
    //在26分、29分、33分执行一次:0 26,29,33 * * * ?
    //每天的0点、13点、18点、21点都执行一次:0 0 0,13,18,21 * * ?
    //在线生成 http://www.pdtools.net/tools/becron.jsp
    @Scheduled(cron = “0 */10 * * * ?”)
    public void test(){
    logger.info(“test当前时间为”+DateUtil.dateToStrLong(new Date())+",时间戳:"+System.currentTimeMillis());
    }

    @Scheduled(cron = “0 */10 * * * ?”)
    public void test1(){
    logger.info(“test1当前时间为”+DateUtil.dateToStrLong(new Date())+",时间戳:"+System.currentTimeMillis());
    }
    }
    import java.util.concurrent.Executors;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;

/**

  • spring boot 多线程并发定时任务

  •  所有的任务都在同一个线程池但不同线程中完成
    
  • @ClassName::ScheduleConfig

  • @author leon

  • @createDate 2018年10月15日 下午4:02:38

  • @version v1.0

  • @classRemarks TODO
    */
    @Configuration
    public class ScheduleConfig implements SchedulingConfigurer {

    @Autowired
    ThreadPoolTaskExecutor threadPoolTaskExecutor;

    @Override
    public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
    taskRegistrar.setScheduler(Executors.newScheduledThreadPool(8,threadPoolTaskExecutor));
    }
    }

部署流程 https://pan.quark.cn/s/a4b39357ea24 在Spring Boot应用中,我们可以利用各种工具来实现并发和动态定时任务。 我们要了解定时任务的基本概念,它是指在特定时间点或按照一定规律自动执行的程序逻辑。 在Java开发领域,有多种方法可以实现定时任务,这些方法在不同场景下各有优缺点。 1. **JDK自带的定时工具**: - `Timer`:是JDK提供的一个基础定时调度类,适合简单的按频率执行的任务,功能较为有限。 - `ScheduledExecutorService`:基于线程池设计,支持并发执行任务,适合处理并发需求,但调度功能较弱。 2. **Spring Task**: - Spring提供的任务调度工具,支持注解配置和Cron表达式,简洁易用,功能相对较全,可以满足大部分日常需求。 3. **Quartz**: - 一个功能强大的第三方调度器,支持复杂的定时策略和分布式调度,但配置相对复杂。 4. **分布式定时任务解决方案**: - 当面临分布式集群环境时,如需解决任务多次执行和单点故障问题,可以采用Quartz、elastic-job、xxl-job或Saturn等分布式任务调度系统。 它们能实现任务的高可用性和分片执行。 **Spring Task在Spring Boot中的实现**:- 使用`@EnableScheduling`注解启动定时任务支持。 - 使用`@Scheduled`注解定义任务,支持cron、fixedRate、fixedDelay等定时策略。 然而,Spring Task的默认实现是单线程执行的,这意味着所有任务会按照顺序执行,这可能导致执行效率不高,尤其是在有阻塞任务时,可能会影响其他任务的执...
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值