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