转自 :http://my.oschina.net/EugeneQiu/blog/265040?p=1
很多人在使用spring的Quartz配置定时任务时,会发现每次定时时间到达后,指定的定时方法同时执行了两次。
(P.S. 本人项目用的Jfinal搭建,同样出现该问题。)
解决quartz定时任务被触发两次的问题:
其中<Host/>告诉tomcat,在启动的时候加载webapps下的所有项目工程文件,<Context/>又让tomcat再加载了一遍(一般情况下配置<Context/>,主要是由于想域名访问时将工程名去掉的原因配置),这种情况下会导致工程中的quartz定时被两次触发,执行两次。
<Host/>里面的改成 autoDeploy="false" deployOnStartup="false" 这样就可以避免tomcat服务器中自启动导致quartz定时任务被触发两次。(当然还可以按需求修改,毕竟每个项目的需求都不尽相同。)
1. autodeploy属性值设置为false,如果此项设为true,表示Tomcat服务处于运行状态时,能够监测appBase下的文件,如果有新有web应用加入进来,会自运发布这个WEB应用,设成false就不会。
2.增加deployOnStartup="false",表示Tomcat服务器启动时, 不会自动发布appBase目录下所有的Web应用。
这样的话,在tomcat启动时,不会自动发布appBase下的应用,启动后也不会自动发步appBase下的应用。
<Host name="localhost" appBase="webapps"
unpackWARs="true" autoDeploy="false" deployOnStartup="false">
<Context path="" docBase="/eugeneqiu_test" debug="0" reloadable="true"/>
</Host>
当然网络上也有另外一种配置方法:
就是在<Context/>里面修改 docBase加上“/webapps”路径,当然项目访问路径就变为http://localhost/eugeneqiu_test 了,看见项目名实在是难受,我就没用这方法了。
<Host name="localhost" appBase="webapps"
unpackWARs="true" autoDeploy="true">
<Context path="" docBase="/webapps/eugeneqiu_test" debug="0" reloadable="true"/>
</Host>
解决quartz定时任务被触发两次的问题
方法1、通过改TOMCAT的配置文件server.xml配置
方法2、先把quartz配置信息提取出来,单独存成一个文件,比如applicationContext-quartz.xml 然后修改web.xml,让web容器启动时,可以加载该文件 。(此文没对此方法进行深入探究)
附带:quartz的测试用例
java测试用例代码1:
package com.eugeneqiu_test.test;
import org.quartz.*;
import org.quartz.impl.StdSchedulerFactory;
/**
* 每5秒自动运行
* @author EugeneQiu
*/
public class Every5s{
public static void orderEveryhour(String[] args) throws SchedulerException {
SchedulerFactory schedulerFactory = new StdSchedulerFactory();
Scheduler scheduler = schedulerFactory.getScheduler();
JobDetail jobDetail = JobBuilder.newJob()
.ofType(Test.class)
.usingJobData("Test1","Quartz")
.withIdentity("Test1","Group1")
.build();//通过JobBuilder构建JobDetailImpl 实例,也可以直接new JobDetailImpl
Trigger trigger = TriggerBuilder.newTrigger()
.withSchedule(CronScheduleBuilder.cronSchedule("0/5 * * * * ?")) //还有更多时间格式,详情可以百度一下
.forJob("Test1","Group1") //Trigger找到对应的名称为Test1组为Group1的Job,如果不存在则会在执行scheduler.scheduleJob(jobDetail,trigger);报错
.build();//通过TriggerBuilder构建CronTriggerImpl实例,也可以直接new CronTriggerImpl
scheduler.scheduleJob(jobDetail,trigger);//任务每5秒触发一次
scheduler.start();
}
}
@Override
public void execute(JobExecutionContext arg0) throws JobExecutionException {
System.out.println("Hello 要开始自动运行喽~~~~~~" );
//testSomething(); //这里可以写一下想定时运行的方法
System.out.println("Hello 自动运行结束~~~~~");
}