一、问题描述
在SpringBoot2.x 的项目上写了个定时任务,每天凌晨1点执行,那么令人费解的是:定时任务在服务器上总是执行两次,这个bug就闹大了,但是在本机调试始终只执行一次,查了很多原因都说是实例化bean两次的问题,但是看了很多解决方案都没有解决…
如下面代码块:
@Scheduled(cron = "0 0 1 * * ?") //每天凌晨一点执行
public void testTask() {
LOGGER.info("进入testTask调度服务,{}", new Date());
//执行业务操作
......
}
输出日志如下:
[2019-02-26 01:00:00.144] [level: INFO] [Thread: pool-16-thread-1] [ Class:com.xxxx.xxxx.task. testTask >> Method: policyWorkNotesYesterdayTask:34 ]
INFO:进入testTask调度服务,Tue Feb 26 01:00:00 UTC 2019
[2019-02-26 01:00:00.155] [level: INFO] [Thread: pool-16-thread-1] [ Class:com.xxxx.xxxx.task. testTask >> Method: testTask:83 ]
INFO:进入testTask调度服务,Tue Feb 26 01:00:00 UTC 2019
执行了两次,并且数据库增加了相同的两次业务记录。
二、问题原因调试分析
问题排查:本地开发和测试都没有问题,说明代码是没有问题的,唯一的差别是本地springboot启动用的内置tomcat容器,服务器上是发布到tomcat下 Webapps 目录,那么问题最有可能出现在这里!
三、Tomcat运行原理分析
Tomcat的Webapps目录是Tomcat默认的应用目录,当服务器启动时,会加载所有这个目录下的应用。也可以将JSP程序打包成一个war包放在目录下,服务器会自动解开这个war包,并在这个目录下生成一个同名的文件夹。一个war包就是有特性格式的jar包,它是将一个Web程序的所有内容进行压缩得到。
项目部署在服务器tomcat的webapps目录里,导致项目被tomcat初始化了2次,部署成功了2次,一次访问路径是项目名的,一次访问路径是/。而本地的开发工具项目部署在wptwebapps不在webapps里,webapps为tomcat默认目录,里面的war包会自动被解压,项目会自动被部署。
五、解决方案
在Tomcat/conf/server.xml下面找到Host这行代码:
<Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="true">
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="localhost_access_log" suffix=".txt"
pattern="%h %l %u %t "%r" %s %b" />
</Host>
配置Tomcat启动项目路径为外部路径(例如项目名为TaskService,在Tomcat目录下创建Task目录),修改配置如下:
<Host name="localhost" appBase="" unpackWARs="true" autoDeploy="true">
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="localhost_access_log" suffix=".txt"
pattern="%h %l %u %t "%r" %s %b" />
<Context docBase="/usr/local/src/tomcat/Task" path="/" reloadable="true" />
</Host>
好了,赶紧重启Tomcat 试试吧!