背景

昨天,咱们的《常识星球:Java技术栈》外面有粉丝向我发问:

问题大略就是:

Spring Boot 定时工作开启后,怎么符合条件主动进行?

过后我有空,尽管曾经给出了参考答案,但可能还有一些细节中央要留神的,另外,我也感觉这个问题特地有意思,当初特地拿进去整顿下,分享下给大家。

1、自定义任务调度

首先笼罩 TaskSchedulingAutoConfiguration 主动配置类外面的 ThreadPoolTaskScheduler Bean:

/** * 自定义任务调度 * 公众号:Java技术栈 */@Data@Componentclass CustomTaskScheduler extends ThreadPoolTaskScheduler {    private Map<Object, ScheduledFuture<?>> scheduledTasks = new IdentityHashMap<>();    @Override    public ScheduledFuture<?> schedule(Runnable task, Trigger trigger) {        ScheduledFuture<?> future = super.schedule(task, trigger);        this.putScheduledTasks(task, future);        return future;    }    @Override    public ScheduledFuture<?> scheduleAtFixedRate(Runnable task, long period) {        ScheduledFuture<?> future = super.scheduleAtFixedRate(task, period);        this.putScheduledTasks(task, future);        return future;    }    @Override    public ScheduledFuture<?> scheduleAtFixedRate(Runnable task, Date startTime, long period) {        ScheduledFuture<?> future = super.scheduleAtFixedRate(task, startTime, period);        this.putScheduledTasks(task, future);        return future;    }    private void putScheduledTasks(Runnable task, ScheduledFuture<?> future) {        ScheduledMethodRunnable runnable = (ScheduledMethodRunnable) task;        scheduledTasks.put(runnable.getTarget(), future);    }        // 重写所有 schedule* 办法...}

Spring Boot 根底就不介绍了,举荐下这个实战教程:

https://github.com/javastacks...

重写所有 schedule* 办法...

因为要进行一个工作,就必须调用 ScheduledFuture -> Future 接口中的 cancel 办法。

所以,思路就是在工作执行的时候,把工作所在的实例 Bean 和工作启动后的 ScheduledFuture 保护到一个 Map 外面,而后须要进行的时候,从 Map 外面取出来,再进行 cancel 进行即可。

2、按条件主动进行工作

新建一个每 3 秒执行一次的工作:

/** * 按条件主动进行工作 * 公众号:Java技术栈 */@Slf4j@Componentpublic class AutoStopTask {    @Autowired    private CustomTaskScheduler customTaskScheduler;    private int count;    @Scheduled(cron = "*/3 * * * * *")    public void printTask() {        log.info("公众号Java技术栈,工作执行次数:{}", count + 1);        count++;        // 执行3次后主动进行        if (count >= 3) {            log.info("工作已执行指定次数,当初主动进行");            boolean cancelled = customTaskScheduler.getScheduledTasks().get(this).cancel(true);                        // 进行后再次启动            if (cancelled) {                count = 0;                ScheduledMethodRunnable runnable = new ScheduledMethodRunnable(this, ReflectionUtils.findMethod(this.getClass(), "printTask"));                customTaskScheduler.schedule(runnable, new CronTrigger("*/3 * * * * *"));            }        }    }}

这里是统计执行,当执行次数超过 3 次时就主动进行。如果须要再次启动,下面也提供了参数代码。

须要留神的是,自定义调度外面绑定的是实例 Bean 和 Future 的关系,所以仅限 Bean 中的单个工作,如果一个 Bean 保护了多个工作,最初一个工作的启动就会笼罩之前的。

如果要保护 Bean 中的多个工作,主动进行该怎么做呢?

答案就是把工作的办法名和 Future 关联起来:

/scheduledMethodTasks.put(runnable.getMethod(), future);

取的的依据以后的办法名取就行了,这里是办法名,也可是类名+办法名+参数,避免反复。

对于这个问题,你还有哪些实现计划呢?欢送留言分享!

总结

本文残缺示例代码曾经上传到 Github:

https://github.com/javastacks...

如果你感兴趣的话,能够 Star 学习,后续会继续更新。

最初打一波咱们星球的广告:

  • 如果你也有许多疑难问题、纳闷无从解答,能够退出咱们的《常识星球:Java技术栈》,只有栈长我晓得的我都会解答,还有 2500+ 球友也在;
  • 如果你学习没有方向、工作没有晋升,《常识星球:Java技术栈》外面也积淀了大量技术常识、学习材料、面试题、简历模板等,退出就值回门票;

星球原价是 199 的,当初是流动优惠价 159 元,正是退出的好时候,过段时间必定会复原原价的。

公众号、微信下面好友太多,不可能一一答复每个人的问题,工夫精力不容许啊(答复是情分,不答复是任务),所以我就创立了《常识星球:Java技术栈》,常识付费,当初就变成了一种责任了,栈长是实打实的纯技术人,不玩套路,也从不玩虚的,咱们的星球相对物超所值,欢送退出一起学习吧。

最初,如果你想关注和学习最新、最支流的 Java 技术,能够继续关注公众号Java技术栈,公众号第一工夫推送。

版权申明: 本文系公众号 "Java技术栈" 原创,转载、援用本文内容请注明出处,剽窃、洗稿一律投诉侵权,后果自负,并保留追究其法律责任的权力。

近期热文举荐:

1.1,000+ 道 Java面试题及答案整顿(2022最新版)

2.劲爆!Java 协程要来了。。。

3.Spring Boot 2.x 教程,太全了!

4.别再写满屏的爆爆爆炸类了,试试装璜器模式,这才是优雅的形式!!

5.《Java开发手册(嵩山版)》最新公布,速速下载!

感觉不错,别忘了顺手点赞+转发哦!