最近几天项目里的定时器常常发生异常,比如:
1、修改linux系统时间时,定时任务全挂掉不动了。
2、在页面里面动态修改任务时间,常常不执行了。
下面是一些网友的资料,暂时收集放着,有时间将去验证和修改自己系统的问题。
近日碰到一位友人提出的一个问题,如何动态的改变Quartz的调度作业的时间。比如,由每10分钟执行一次改为每5分钟执行一次。个人认为这种需求应该 通过某种方式来规避,或者选用其他的技术框架,因为动态改变Quartz的调度时间完全失去了使用Quartz的意义。本人在使用Quartz是基于 Spring来配置的,而朋友的项目中不能使用SPring框架,这就需要直接基于Quartz编程。工作之余,写了个例子:
Quartz的管理类
- public class QuartzManage{
- private static SchedulerFactorysf= new StdSchedulerFactory();
- private static StringJOB_GROUP_NAME= "group" ;
- private static StringTRIGGER_GROUP_NAME= "trigger" ;
- public static void startJob(StringjobName,Jobjob,Stringtime)
- throws SchedulerException,ParseException{
- Schedulersched=sf.getScheduler();
- JobDetailjobDetail=new JobDetail();
- jobDetail.setName(jobName);
- jobDetail.setGroup(JOB_GROUP_NAME);
- jobDetail.setJobClass(job.getClass());
- CronTriggertrigger=new CronTrigger(jobName,TRIGGER_GROUP_NAME);
- trigger.setCronExpression(time);
- sched.scheduleJob(jobDetail,trigger);
- if (!sched.isShutdown()){
- sched.start();
- }
- }
- /**
- *从Scheduler移除当前的Job,修改Trigger
- *
- *@paramjobDetail
- *@paramtime
- *@throwsSchedulerException
- *@throwsParseException
- */
- public static void modifyJobTime(JobDetailjobDetail,Stringtime)
- throws SchedulerException,ParseException{
- Schedulersched=sf.getScheduler();
- Triggertrigger=sched.getTrigger(jobDetail.getName(),
- TRIGGER_GROUP_NAME);
- if (trigger!= null ){
- CronTriggerct=(CronTrigger)trigger;
- //移除当前进程的Job
- sched.deleteJob(jobDetail.getName(),jobDetail.getGroup());
- //修改Trigger
- ct.setCronExpression(time);
- System.out.println("CronTriggergetName" +ct.getJobName());
- //重新调度jobDetail
- sched.scheduleJob(jobDetail,ct);
- }
- }
- }
Job任务:
- public class JobTest implements Job{
- static int a= 0 ;
- @Override
- public void execute(JobExecutionContextcontext)
- throws JobExecutionException{
- a+=1 ;
- System.out.println("test++++++++++++++++++++++a=" +a);
- if (a== 4 ){
- try {
- QuartzManage.modifyJobTime(context.getJobDetail(),
- "0/10****?" );
- }catch (SchedulerExceptione){
- e.printStackTrace();
- }catch (ParseExceptione){
- e.printStackTrace();
- }
- }
- }
- }
启动线程执行调度:
- public class QuartzTest{
- public static void main(String[]args) throws SchedulerException,
- ParseException{
- /*
- *此进程为主进程,触发了quartz对Job的调度因此启动Job之后,在该进程修改调度,是没有效果的
- */
- JobTestjob=new JobTest();
- QuartzManage.startJob("ming" ,job, "0/2****?" );
- }
- }
好多人的思路是在启动的主线程内去改变调度的时间,简单的分析就可发现,主线程启动之后就会按照调度时间去运行Job,不会返回主线程再去加载调度时间,只是起到了触发调度的操作。因此要进行动态的修改调度时间,需要在Job任务里,动态改变当前线程的调度计划。
测试代码,开始时按每2秒执行一次打印a,当a打印4次以后,按照每10秒一次执行。虽然代码测试成功,本人还有疑惑。
- //移除当前进程的Job
- sched.deleteJob(jobDetail.getName(),jobDetail.getGroup());
先移除当前的Job任务,在按照新的调度时间加入新的Job,虽然可以实现动态的改变,不知道是否带来了其他的问题。
欢迎大家批评指正共同测试、验证!