quartz 在WEB中应用小结

前一个项目曾经用到过quartz,没好好总结一下,现在的项目中又用到了,害得我又花了查了两天查资料、做测试!现在OK了,系统目前运行正常,做个小结。

1. 基本配置

官网上有quartz的所有版本,下一个,把quartz-all-*.jar(根据需要)添加到lib目录中,在classpath目录添加quartz.properties属性文件,接下来你就可以进行开发了。
这是quartz.properties的基本配置。

#============================================================================
# Configure Main Scheduler Properties  
#============================================================================

org.quartz.scheduler.instanceName = TestScheduler
org.quartz.scheduler.instanceId = AUTO

#============================================================================
# Configure ThreadPool  
#============================================================================

org.quartz.threadPool.class = org.quartz.simpl.SimpleThreadPool
org.quartz.threadPool.threadCount = 3
org.quartz.threadPool.threadPriority = 5

#============================================================================
# Configure JobStore  
#============================================================================

org.quartz.jobStore.misfireThreshold = 60000

org.quartz.jobStore.class = org.quartz.simpl.RAMJobStore



2. web基本应用

quartz在web项目中有多种应用方式,和项目使用的其他技术也有关系。

添加一个servlet,将它配置到web.xml中:

<servlet>
		<servlet-name>QuartzStartupServlet</servlet-name>
		<servlet-class>com.StartupServlet</servlet-class>
		<load-on-startup>1</load-on-startup>
</servlet>


这样你就可以在这个servlet中,在服务器加载的时候同时加载定时任务:

public void init(ServletConfig cfg) throws javax.servlet.ServletException
	{
		log.info("server start!");
		initScheduler(cfg);
	}

	protected void initScheduler(ServletConfig cfg)
	{
		try
		{
			SchedulerFactory sf = new StdSchedulerFactory();
			Scheduler scheduler = sf.getScheduler();
			SimpleExample simpleExample = new SimpleExample(scheduler);
			simpleExample.run();
		} catch (Exception e)
		{
			log.error(e);
		}
	}


任务定义和加载在quartz的示例中有,这里不多费口舌了。

3. xml形式的任务配置

xml形式的应用,需要使用quartz的xml支持插件,只需有实现Job接口的任务就可以;这时需要在web.xml中添加默认的servlet配置:

<servlet>
		<display-name>Quartz Initializer Servlet</display-name>
		<servlet-name>QuartzInitializer</servlet-name>
		<servlet-class>
			org.quartz.ee.servlet.QuartzInitializerServlet
		</servlet-class>
		<load-on-startup>1</load-on-startup>
	</servlet>


在quartz.properties中添加xml配置需要的参数:

#============================================================================
# Configure Plugins 
#============================================================================

org.quartz.plugin.triggHistory.class = org.quartz.plugins.history.LoggingJobHistoryPlugin

org.quartz.plugin.jobInitializer.class = org.quartz.plugins.xml.JobInitializationPlugin
org.quartz.plugin.jobInitializer.fileNames = jobs.xml
org.quartz.plugin.jobInitializer.overWriteExistingJobs = true
org.quartz.plugin.jobInitializer.failOnFileNotFound = true
org.quartz.plugin.jobInitializer.scanInterval = 10
org.quartz.plugin.jobInitializer.wrapInUserTransaction = false

上面配置中的fileNames是指向的classpath根目录。

在xml文件中定义任务名称和触发器,例如:
<?xml version="1.0" encoding="UTF-8"?>
<quartz>
	<job>
		<job-detail>
			<name>dailybalance</name>
			<group>balance</group>
			<job-class>com.SimpleJob</job-class>
		</job-detail>
		<trigger>
			<cron>
				<name>dailybalancetrigger</name>
				<group>balance</group>
				<job-name>dailybalance</job-name>
				<job-group>balance</job-group>
				<cron-expression>0/50 * * * * ?</cron-expression>
			</cron>
		</trigger>
	</job>
</quartz>

这样就OK了!几种配置可以同时应用,不会冲突!

4. quartz的动态时间管理

有时需要在不停止web服务的情况下修改任务的执行时间格式,网上的资料大多是在使用spring框架支持时如何动态管理时间格式的介绍。在使用spring支持时可以,不使用spring时更加简单。

动态时间管理的关键是要共享调度管理器对象Scheduler,我们可以在最初加载任务调度的时候将它添加到servlet上下文中。
SchedulerFactory sf = new StdSchedulerFactory();
Scheduler scheduler = sf.getScheduler();
......
......
cfg.getServletContext().setAttribute(SCHEDULER_KEY, scheduler);
scheduler.start();

这样,我们就可以在需要修改时间格式的地方使用了!下面的ctt变量就是一个ServletContext对象。通过Scheduler 的getTrigger方法取到触发器,用触发器对象的getCronExpression方法可以获得时间格式。之后你可以另外创建一个同名的触发器,也可以修改现有触发器,再让调度管理器重新加载到内存就可以了!
/**
	 * reset quartz job time
	 */
	public void reSetLotteryLoseBiddingTime(int lotteryType, Date date)
	{
		scheduler = (Scheduler) this.ctt.getAttribute(TIMING_WORK_SCHEDULER);
		try
		{
			String lotteryName = "";
			switch (lotteryType)
			{
			case 1://
				lotteryName = "ssq";
				break;
			case 2:// 
			{
				lotteryName = "3D";
			}
				break;
			case 3:// 
			{
				lotteryName = "qlc";
			}
				break;
			case 4:// 
			{
				lotteryName = "23x5";
			}
				break;
			default:
				break;
			}

			loseBiddingJob = scheduler.getJobDetail("loseBiddingJob_" + lotteryName, "loseBiddingGroup");
			loseBiddingTrigger = (CronTrigger) scheduler.getTrigger("loseBiddingTrigger_" + lotteryName,
					"loseBiddingGroup");
			String time = loseBiddingTrigger.getCronExpression();
			if (!time.equals(this.timeFormat(date)))
			{
				loseBiddingTrigger = new CronTrigger("loseBiddingTrigger_" + lotteryName, "loseBiddingGroup",
						"loseBiddingJob_" + lotteryName, "loseBiddingGroup", this.timeFormat(date));
				Date ft = scheduler.rescheduleJob("loseBiddingTrigger_" + lotteryName, "loseBiddingGroup",
						loseBiddingTrigger);
				log.info(loseBiddingJob.getFullName() + " has been scheduled to run at: " + ft
						+ " and excute based on expression: " + loseBiddingTrigger.getCronExpression());
			}
		} catch (Exception e)
		{
			log.error(e);
		}
	}



5.相关问题

配置完成后,定时任务不加载执行,在日志中报错:
java.lang.NoSuchMethodError: org.apache.commons.collections.SetUtils.orderedSet(Ljava/util/Set;)Ljava/util/Set;
这个是commons-collections.jar版本过低的原因,更换个新版本的就可以了!

还有,为了防止开发人员在本地测试时,执行定时任务造成数据冲突,需要对主机服务器Ip进行判断,只有指定的服务器上才可以执行定时任务!
public boolean compareService(String thisIp)
{
	Properties properties = this.getProperties();
	String serviceIp = "";
	try
	{
		if (properties != null)
		{
			serviceIp = properties.getProperty("timing.work.serviceIp");
			if (serviceIp != null && serviceIp.equals(thisIp))
				return true;
		}
	} catch (Exception e)
	{
		log.error(e);
	}
	return false;
}




quartz还可以对数据库操作,还有JobStore等等,以后应用到再说!哈!附带一份《quartz开发指南》!

你可能感兴趣的:(spring,Web,应用服务器,quartz,配置管理)