SpringBoot踩坑日记-定时任务不定时了?

问题描述br/>springboot定时任务用起来大家应该都会用,加两注解,加点配置就可以运行。但是如果仅仅处在应用层面的话,有很多内在的问题开发中可能难以察觉。话不多说,我先用一种极度夸张的手法,描述一下遇到的一个问题。
@Component
public class ScheduleTest {
@Scheduled(initialDelay = 1000,fixedRate = 2*1000)
public void test_a(){
System.out.println("123");
}

@Scheduled(initialDelay = 2*1000,fixedRate = 2*1000)
public void test_b(){
    while (true){
        try {
            Thread.sleep(2*1000);
            System.out.println("456");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

}
复制代码上面代码是一个项目中的两个定时任务,test_a是正常的方法,test_b是发生异常的方法,为了凸显异常,我搞了个死循环。
在这种情况下,使用默认的定时任务配置运行,会发生什么现象呢?试试看就知道了,定时任务一直在方法b中循环着,方法a永远执行不到!!!

问题原因
查看源代码后发现SpringBoot源码解析-Scheduled定时器的原理,springboot中,默认的定时任务线程池是只有一个线程的,所以如果在一堆定时任务中,有一个发生了延时或者死循环之类的异常,很大可能会影响到其他的定时任务。
解决方案
既然问题出在线程池数量上,那么为了让各个任务之间不会互相干扰,那就配置相应的线程池就好了。
方案一 异步执行
@Scheduled(initialDelay = 1000,fixedRate = 2*1000)br/>@Async
public void test_a(){
System.out.println("123");
}

@Scheduled(initialDelay = 2*1000,fixedRate = 2*1000)
@Async
public void test_b(){
    while (true){
        try {
            Thread.sleep(2*1000);
            System.out.println("456");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

复制代码既然在单线程中因为一个任务卡住而影响到其他任务,那么把这个任务异步执行,问题就解决啦。
方案二 自定义定时任务线程池数量
br/>@Configuration
public class GlobalConfiguration {

@Bean
public TaskScheduler schedule(){
    return new ConcurrentTaskScheduler(new ScheduledThreadPoolExecutor(2));
}

}
复制代码既然单线程会互相干扰,那么分配足够的线程,让他们各自分开运行,也是可以解决的。
两种方案对比
两种方案都可以解决各个任务之间互相干扰的问题,但是需要根据实际情况选择合适的。我们就以上面出现死循环的代码来分析。
如果在定时任务中真的发生了死循环,那么使用异步执行则会带来灾难性的后果。因为在定时任务这个线程中,每次任务执行完毕后,他会计算下次时间,再次添加一个任务进入异步线程池。而添加进异步线程池的任务因为死循环而一直占用着线程资源。随着时间的增加异步线程池的所有线程资源都会被死循环的任务占据,导致其他服务全部阻塞。
而使用自定义定时任务线程池则会好一点,因为只有当任务执行完成后,才会计算时间,在执行下次任务。虽然因为死循环任务一直在执行,但是也顶多占据一个线程的资源,不至于更大范围的影响。

SpringBoot踩坑日记-定时任务不定时了?

原文地址:https://blog.51cto.com/13942046/2379626

时间: 2024-10-20 20:22:21

SpringBoot踩坑日记-定时任务不定时了?的相关文章

米忽悠踩坑日记-1

米忽悠踩坑日记-1            --知不足,而后进 进入米哈游差不多一个半月了,就以刚刚上线的某个任务作为节点写一篇踩坑日记吧. 1.安全意识,尽量考虑到玩家各种奇奇怪怪的操作以及有可能想刷道具的行为. 2.日志方面,记录玩家的每一步操作,成功或者失败,需要记录清楚,uid,region以及其他的信息,如奖励的ID,更新一次游戏玩家数据也记录 3.在写代码时候不要总想着先实现逻辑再来优化结构,因为一个小任务的代码量不一定少,而且任务排的很紧,如果不从一开始就保持良好的结构自己看起来简直

vue 组件传值踩坑日记 1

今天在用平时很少用到的传值方式,是V2.4以后新加入属性$attrs $listener 以及inheritAttrs. 总结:这样形式的代码适合套娃模式的组件传递,却不适合兄弟组件的传值,那样的传值方式,需要创建一个事件总线,说白了就是新new一个空的vue,详见,我的第二篇日志<vue 组件传值踩坑日记 2> 废话不多说,直接上代码,大家可以边看注释边测试一下效果吧 <!DOCTYPE html> <html lang="en"> <hea

AI相关 TensorFlow -卷积神经网络 踩坑日记之一

上次写完粗浅的BP算法 介绍 本来应该继续把 卷积神经网络算法写一下的 但是最近一直在踩 TensorFlow的坑.所以就先跳过算法介绍直接来应用场景,原谅我吧. TensorFlow 介绍 TF是google开源出来的人工智能库,由python语言写的 官网地址:http://www.tensorflow.org/   请用科学上网访问 中文地址:http://www.tensorfly.cn/ 当然还有其他AI库,不过大多数都是由python 写的 .net 的AI库叫 Accord.net

人工智能(AI)库TensorFlow 踩坑日记之二

上次 踩坑日志之一 遗留的问题终于解决了,所以作者(也就是我)终于有脸出来写第二篇了. 首先还是贴上 卷积算法的示例代码地址 :https://github.com/tensorflow/models   这个库里面主要是一些常用的模型用tensorflow实现之后的代码.其中我用的是 models/tree/master/tutorials/image/cifar10 这个示例,上一篇也大致讲过了. 关于上次遇到问题是: 虽然训练了很多次,但是每次实际去用时都是相同的结果.这个问题主要原因是

quick-cocos2d-x 踩坑日记(一)------ NodeEx 与 shortcodes

今天在模仿一个demo,想自己去完善一些功能,自己写的时候就遇见了一个坑...先看代码: local label = cc.ui.UILabel.new({     UILabelType = 1,     text = "Max Score 10",     font = "fonts/font-issue1343.fnt",     x = boardSize.width / 2,     y = boardSize.height * 0.4,     }):a

cocos2dx 3.3rc0 踩坑日记(一)------ 弱联网 Curl

这两天看了视频学习了下弱联网技术,用的是Curl,具体使用方法我就多说了,可以参考官方样例和下面的文章,解释的很清楚. [Curl (libcurl) 开发 之一]Cocos2dx之libcurl(curl_easy)的编程教程(帮助手册)! CURL使用2 我要说的我又踩到一个坑...为什么是又呢...天生就是来踩坑的... 先说下使用之前需要加入curl.h的路径$(EngineRoot)external\curl\include\win32和链接库libcurl_imp.lib 如图所示:

cocos creator踩坑日记(持续更新中...)

踩坑一 问题:项目在构建成Web Mobile后运行在浏览器和微信中,点击页面任何地方都会导致自动全屏 解决:在构建之后的main.js中,去掉 cc.view.enableAutoFullScreen(true) 踩坑二 问题:精灵使用move to后路径上会留下一些线条 原因:脏矩形出问题了 解决: Canvas模式下才有脏矩形,使用这个方法之前要判断 if (cc._renderType === cc.game.RENDER_TYPE_CANVAS) { cc.renderer.enabl

quick cocos2dx 3.3rc1 踩坑日记(二)------ 声音引擎

昨天在群里有人说setSoundsVolume和setMusicVolume设置声音大小无效...我记得cocos2dx是没实现内容的,就贸贸然回答说,是这你的没用,但群里有人说是有用的,我就去虚心请教,大神人很好,回答了我很多问题,告诉我说实现了,而且是针对不同的平台.实现的文件位置,如下图: 我先测试setMusicVolume,在win32下setMusicVolume是无效的,在Android上是可以的. 但在测试setSoundsVolume时,就遇见问题了,当调用audio.play

html2canvas踩坑日记

近日要开发一个能将生成的二维码另存为图片的功能(该图片呢,肯定不止一个二维码,还包括背景.文字等其他元素),首先呢,就想到了html2canvas,随便一百度就是各种踩坑日志,我也随一下大流,记录本人在开发过程中遇到的坑. 1.基本用法: 在html2canvas上找到了它的基本用法以及压缩包 <div id="capture" style="padding: 10px; background: #f5da55"> <h4 style="