springboot 使用 @Scheduled 注解实现任务调度 以及相关踩坑记录

关于spring boot 实现任务调度的方法有很多种,这里不再赘述。

首先pom只需引入

<dependency>
    <groupId>org.quartz-scheduler</groupId>
    <artifactId>quartz</artifactId>
    <version>2.2.1</version>
</dependency>

然后application开启@EnableScheduling

@SpringBootApplication
@EnableAsync
@EnableScheduling
public class Application extends SpringBootServletInitializer {

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
        return application.sources(Application.class);
    }

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


}

之后用嘛就很简单了

@Component
@Configurable
public class TestONE {
    private Logger logger = (Logger) LoggerFactory.getLogger(TestONE.class);

    @Scheduled(cron = "0/5 * * * * ? ")
//    @Async
    public  void test1() {
        logger.error("任务1--开始");
        try {
            Thread.sleep(20000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        logger.error("任务1---结束");
    }

========分隔线=========

首先大致说下遇到过的问题

Q1:启动后修改本地时间但是调度任务不再执行

  A:之前大致看了一下这块的介绍,其实启动之后虚拟机内维持一个时间计数器,这个是启动时与系统时间同步过的,当让会有一个专门管理调度任务的,按照虚拟机时间以及系统时间核对(如果这出现不一致肯定不执行了),如果一致那么也满足cron规则就开启执行。

Q2:出现漏执行的情况

A:我目前了解到其实spring的@Scheduled默认是单线程的,也就是说一次只能执行一个定时任务,如果一但任务是耗时的那么后面的自动进入线程的挂起等待队列,当然这个队列也是有大小限制,如果执行中任务特别耗时挂起的超过限制那么后面的任务都不会执行。

        很多大企是很注重调度任务的可用性,因为毕竟对于任务调度这个事情来说里面涉及到的业务肯定都是有价值甚至很重要的。特别公司发展的一定程度调度任务的重要性非常的高。所以如果还是单节点的架构的调度的话建议还是上高可用的调度吧,那么关于高可用的调度任务网上还是有好几个解决方案以供选择,其实最核心的一点哪怕是多节点调度也是单一机器去执行调度任务,也就要求做到分布式锁,一般来讲redis 或者 zookeeper就可以很好的实现这块。当然还有一个很重要的事儿就是关于调度任务失败的重调。个人推荐elastic-job这个组件,可以自行研究对比后选择适合自己公司业务的。

        针对以上调度线程是单线程问题其实也可以通过,设置执行线程数目,建议开启的线程池核心线程不要过高,毕竟cpu也就那么多个核心,线上项目又肯定不是自己一个的。何况还有维护系统其他组件等的线程。个人感觉视自己项目里调度数量和执行时间决定。单一时间段内不会出现跑满即可。

/**
 * @author qiaolu
 */
@Configuration
public class ScheduledConfig implements SchedulingConfigurer {

    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        scheduledTaskRegistrar.setScheduler(setTaskExecutors());
    }

    @Bean
    public Executor setTaskExecutors() {
        return newScheduledThreadPool(2);
    }
}

当然针对耗时较长的调度任务其实完成可以这样解决:加上 @Async

spring会把这个方法另开启一个线程去处理,以达到异步处理的效果,这样调度任务也会快速。当然没必要给所有调度都这么加。

@Scheduled(cron = "0/5 * * * * ? ")
@Async
public  void test1() {
    logger.error("任务1--开始");
    try {
        Thread.sleep(20000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    logger.error("任务1---结束");
}

好了 大致就是以上的几点,基本也就满足中小企业遇到的调度问题。当然也可能有描述不准确或者不全面的地方,欢迎指正,哈哈。

转载于:https://my.oschina.net/u/2337666/blog/1929365

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是一个使用 `@Scheduled` 注解实现定时任务的 Spring Boot 示例代码: 首先,在 Spring Boot 项目的 pom.xml 文件中添加以下依赖: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> </dependency> ``` 然后,在一个 Spring Boot 的 Java 类中添加以下代码: ```java import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component; @Component public class ScheduledTasks { @Scheduled(fixedRate = 5000) // 每隔 5 秒执行一次 public void task1() { System.out.println("Task1: " + System.currentTimeMillis()); } @Scheduled(cron = "0 0/1 * * * ?") // 每分钟执行一次 public void task2() { System.out.println("Task2: " + System.currentTimeMillis()); } } ``` 上面代码中,我们使用 `@Scheduled` 注解实现两个定时任务,一个是每隔 5 秒执行一次,另一个是每分钟执行一次。 其中,`@Scheduled` 注解有两个常用的属性: - `fixedRate`:表示任务执行的间隔时间,单位为毫秒。 - `cron`:表示基于 Cron 表达式的定时任务配置,格式为 `second minute hour dayOfMonth month dayOfWeek year`。 最后,启动 Spring Boot 应用程序,即可看到定时任务在后台自动执行的输出结果。 注意:在 Spring Boot 应用程序中使用 `@Scheduled` 注解时,需要在应用程序主类上添加 `@EnableScheduling` 注解,以启用定时任务功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值