最流行的java后臺(tái)框架spring quartz定時(shí)任務(wù)
配置quartz 在spring中需要三個(gè)jar包:
quartz-1.8.5.jar、commons-collections-3.2.1.jar、commons-logging-1.1.jar
首先要配置我們的spring.xml
xmlns 多加下面的內(nèi)容、
xmlns:task="
然后xsi:schemaLocation多加下面的內(nèi)容、 http://www.springframework.org/schema/task 最后是我們的task任務(wù)掃描注解 <task:annotation-driven/> 我的配置掃描位置是:
http://www.springframework.org/schema/task/spring-task-3.1.xsd
<bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor"/>
<context:component-scan base-package="com.test"/>
掃描的是com.test這樣的包下的內(nèi)容、
下面需要接口和實(shí)現(xiàn)(我的這幾個(gè)java文件都是com.test的包下的、)
public interface IMyTestService { public void myTest(); }
@Component //import org.springframework.stereotype.Component; public class MyTestServiceImpl implements IMyTestService { @Scheduled(cron="0/5 * * * * ? ") //每5秒執(zhí)行一次 @Override public void myTest(){ System.out.println("進(jìn)入測(cè)試"); } }
執(zhí)行后控制臺(tái)就會(huì)打印出 進(jìn)入測(cè)試 了
需要注意的幾點(diǎn):
1、spring的@Scheduled注解 需要寫(xiě)在實(shí)現(xiàn)上、
2、 定時(shí)器的任務(wù)方法不能有返回值(如果有返回值,spring初始化的時(shí)候會(huì)告訴你有個(gè)錯(cuò)誤、需要設(shè)定一個(gè)proxytargetclass的某個(gè)值為true、具體就去百度google吧)
3、實(shí)現(xiàn)類上要有組件的注解@Component
剩下的就是corn表達(dá)式了、具體使用以及參數(shù)請(qǐng)百度google、
【秒】 【分】 【時(shí)】 【日】 【月】 【周】 【年】
下面只例出幾個(gè)式子
CRON表達(dá)式 含義
"0 0 12 * * ?" 每天中午十二點(diǎn)觸發(fā)
"0 15 10 ? * *" 每天早上10:15觸發(fā)
"0 15 10 * * ?" 每天早上10:15觸發(fā)
"0 15 10 * * ? *" 每天早上10:15觸發(fā)
"0 15 10 * * ? 2005" 2005年的每天早上10:15觸發(fā)
"0 * 14 * * ?" 每天從下午2點(diǎn)開(kāi)始到2點(diǎn)59分每分鐘一次觸發(fā)
"0 0/5 14 * * ?" 每天從下午2點(diǎn)開(kāi)始到2:55分結(jié)束每5分鐘一次觸發(fā)
"0 0/5 14,18 * * ?" 每天的下午2點(diǎn)至2:55和6點(diǎn)至6點(diǎn)55分兩個(gè)時(shí)間段內(nèi)每5分鐘一次觸發(fā)
"0 0-5 14 * * ?" 每天14:00至14:05每分鐘一次觸發(fā)
"0 10,44 14 ? 3 WED" 三月的每周三的14:10和14:44觸發(fā)
"0 15 10 ? * MON-FRI" 每個(gè)周一、周二、周三、周四、周五的10:15觸發(fā)
有時(shí)候我們的任務(wù)(Job)需要再某些任務(wù)完成之后才能進(jìn)行;例如從舊的數(shù)據(jù)庫(kù)批量導(dǎo)數(shù)據(jù)的時(shí)候;需要現(xiàn)將被其他數(shù)據(jù)依賴的數(shù)據(jù)導(dǎo)入新的數(shù)據(jù)庫(kù);然后再進(jìn)行關(guān)系的導(dǎo)入.。在這種情況下我們就可以使用Quartz的listener來(lái)做文章了。
首先我們寫(xiě)一個(gè)主任務(wù)的類,命名為MainJob;她的作用是作為一系列任務(wù)的開(kāi)始點(diǎn)。
MainJob.java
package jobs; import org.apache.log4j.Logger; import org.quartz.JobExecutionContext; import org.quartz.JobExecutionException; import org.springframework.scheduling.quartz.QuartzJobBean; public class MainJob extends QuartzJobBean { private Logger logger=Logger.getLogger(getClass()); @Override protected void executeInternal(JobExecutionContext arg0) throws JobExecutionException { // TODO Auto-generated method stub logger.debug("Just say hi."); } }
然后我們新建另外一個(gè)任務(wù)(SecondJob)作為后續(xù)任務(wù):
SecondJob.java
package jobs; import org.apache.log4j.Logger; import org.quartz.JobExecutionContext; import org.quartz.JobExecutionException; import org.springframework.scheduling.quartz.QuartzJobBean; public class SecondJob extends QuartzJobBean { private Logger logger=Logger.getLogger(getClass()); @Override protected void executeInternal(JobExecutionContext arg0) throws JobExecutionException { // TODO Auto-generated method stub logger.debug("I'm the second job."); } }
創(chuàng)建一個(gè)TriggerListener,重寫(xiě)其triggerComplete方法,并且添加一些方便spring注入的屬性和方法。
NextJobTriggerListener.java
package listeners; import org.apache.log4j.Logger; import org.quartz.JobDetail; import org.quartz.JobExecutionContext; import org.quartz.Scheduler; import org.quartz.Trigger; import org.quartz.listeners.TriggerListenerSupport; import org.springframework.scheduling.quartz.QuartzJobBean; import org.springframework.scheduling.quartz.SimpleTriggerBean; public class NextJobTriggerListener extends TriggerListenerSupport { private Logger logger=Logger.getLogger(getClass()); private String name; public String getName() { return this.name; } public void setName(String name) { this.name=name; } private SimpleTriggerBean nextTrigger; public void setNextTrigger(SimpleTriggerBean nextTrigger) { this.nextTrigger=nextTrigger; } @Override public void triggerComplete(Trigger trigger, JobExecutionContext context, int code) { try{ Scheduler schduler=context.getScheduler(); JobDetail nextJob=nextTrigger.getJobDetail(); //查找名稱和即將加入的任務(wù)一樣的任務(wù) JobDetail oldJob=schduler.getJobDetail(nextJob.getName(),nextJob.getGroup()); //查找名稱和即將加入的觸發(fā)器一樣的觸發(fā)器 Trigger oldTrigger=schduler.getTrigger(nextTrigger.getName(),nextTrigger.getGroup()); if(oldJob==null&&oldTrigger==null)//同名的任務(wù)和觸發(fā)器都不存在 { logger.debug("inside scheduleJob."+code); schduler.scheduleJob(nextJob,nextTrigger); }else//同名的任務(wù)或觸發(fā)器 { logger.debug("oldJob==null:"+(oldJob==null)); logger.debug("oldTrigger==null:"+(oldTrigger==null)); } super.triggerComplete(trigger, context, code); }catch(Exception e) { e.printStackTrace(); } } }
配置spring 的applicationContext.xml
applicationContext.xml
<?xml version="1.0"encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:jee="http://www.springframework.org/schema/jee" xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans-2.0.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.0.xsd"> <!-- 主任務(wù) --> <bean id="mainJob" class="org.springframework.scheduling.quartz.JobDetailBean"> <!-- 運(yùn)行的類 --> <property name="jobClass"> <value> jobs.MainJob </value> </property> </bean> <!-- 主任務(wù)的監(jiān)聽(tīng)器 --> <bean id="mainTriggerListener" class="listeners.NextJobTriggerListener"> <!-- 下個(gè)觸發(fā)器 --> <property name="nextTrigger" ref="secondTrigger"></property> <!-- 監(jiān)聽(tīng)器名稱 --> <property name="name" value="mainTriggerListener"></property> </bean> <!-- 主任務(wù)的觸發(fā)器 --> <bean id="mainTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean"> <property name="jobDetail"> <!-- 上面創(chuàng)建的任務(wù)調(diào)度對(duì)象 --> <ref bean="mainJob" /> </property> <!-- 啟動(dòng)60秒后執(zhí)行任務(wù)調(diào)度的excute方法 --> <property name="startDelay"> <value> 6000 </value> </property> <!-- 運(yùn)行次數(shù) --> <property name="repeatCount"> <value> </value> </property> <!-- 隔一個(gè)小時(shí)運(yùn)行一次(貌似多余,不寫(xiě)會(huì)報(bào)錯(cuò)) --> <property name="repeatInterval"> <value> 3600000 </value> </property> <property name="triggerListenerNames"> <list> <value> mainTriggerListener </value> </list> </property> </bean> <!-- 后續(xù)任務(wù) --> <bean id="secondJob" class="org.springframework.scheduling.quartz.JobDetailBean"> <!-- 運(yùn)行的類 --> <property name="jobClass"> <value> jobs.SecondJob </value> </property> </bean> <!-- 后續(xù)任務(wù)的觸發(fā)器 --> <bean id="secondTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean"> <property name="jobDetail"> <!-- 上面創(chuàng)建的任務(wù)調(diào)度對(duì)象 --> <ref bean="secondJob" /> </property> <!-- 啟動(dòng)6秒后執(zhí)行任務(wù)調(diào)度的excute方法 --> <property name="startDelay"> <value> 6000 </value> </property> <!-- 運(yùn)行次數(shù) --> <property name="repeatCount"> <value> </value> </property> <!-- 隔一個(gè)小時(shí)運(yùn)行一次(貌似多余,不寫(xiě)會(huì)報(bào)錯(cuò)) --> <property name="repeatInterval"> <!-- <value>3600000</value> --> <value> 6000 </value> </property> </bean> <!-- 任務(wù)調(diào)度工廠類 --> <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean"> <!-- 這一部分的配置不用管 --> <property name="quartzProperties"> <props> <prop key="org.quartz.threadPool.class"> org.quartz.simpl.SimpleThreadPool </prop> <prop key="org.quartz.threadPool.threadCount"> </prop> <prop key="org.quartz.threadPool.threadPriority"> </prop> <prop key="org.quartz.threadPool.threadsInheritContextClassLoaderOfInitializingThread"> true </prop> </props> </property> <!-- 觸發(fā)器,可以放一大堆觸發(fā)器 --> <property name="triggers"> <list> <!-- 在這里加 --> <ref bean="mainTrigger"/> </list> </property> <property name="triggerListeners"> <list> <!-- 觸發(fā)器的監(jiān)聽(tīng)器 --> <ref bean="mainTriggerListener" /> </list> </property> </bean> </beans>
開(kāi)啟服務(wù)器,輸出
DEBUG [ MainJob.executeInternal(14) ] Just say hi. DEBUG [ NextJobTriggerListener.triggerComplete(38) ] inside scheduleJob .3 DEBUG [SecondJob.executeInternal(14)] I'm the second job. DEBUG [ NextJobTriggerListener.triggerComplete(43) ] oldJob==null:false DEBUG [ NextJobTriggerListener.triggerComplete(44) ] oldTrigger== null:false
另外這里一個(gè)任務(wù)只綁定了一個(gè)簡(jiǎn)單的觸發(fā)器,這樣做是為了比較方便地可以檢測(cè)到任務(wù)完成的情況;至于任務(wù)的具體內(nèi)容就任由大家發(fā)揮了。寫(xiě)這篇文章希望能有人在其中獲得啟發(fā)。
- Java定時(shí)任務(wù)的三種實(shí)現(xiàn)方法
- Java定時(shí)任務(wù)的三種實(shí)現(xiàn)方式
- Quartz實(shí)現(xiàn)JAVA定時(shí)任務(wù)的動(dòng)態(tài)配置的方法
- Java定時(shí)任務(wù):利用java Timer類實(shí)現(xiàn)定時(shí)執(zhí)行任務(wù)的功能
- Java實(shí)現(xiàn)終止線程池中正在運(yùn)行的定時(shí)任務(wù)
- java定時(shí)任務(wù)Timer和TimerTask使用詳解
- Java 實(shí)現(xiàn)定時(shí)任務(wù)的三種方法
- java中 spring 定時(shí)任務(wù) 實(shí)現(xiàn)代碼
- 在Java Web項(xiàng)目中添加定時(shí)任務(wù)的方法
- java實(shí)現(xiàn)周期性執(zhí)行(定時(shí)任務(wù))
相關(guān)文章
Spring?Boot日志基礎(chǔ)使用之如何設(shè)置日志級(jí)別
這篇文章主要介紹了Spring?Boot日志基礎(chǔ)使用設(shè)置日志級(jí)別的方法,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09Java?SM2加密相關(guān)實(shí)現(xiàn)與簡(jiǎn)單原理詳解
SM2算法可以用較少的計(jì)算能力提供比RSA算法更高的安全強(qiáng)度,而所需的密鑰長(zhǎng)度卻遠(yuǎn)比RSA算法低,這篇文章主要給大家介紹了關(guān)于Java?SM2加密相關(guān)實(shí)現(xiàn)與簡(jiǎn)單原理的相關(guān)資料,需要的朋友可以參考下2024-01-01Java substring方法實(shí)現(xiàn)原理解析
這篇文章主要介紹了Java substring方法實(shí)現(xiàn)原理解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-05-05spring?boot中spring框架的版本升級(jí)圖文教程
Spring Boot是一款基于Spring框架的快速開(kāi)發(fā)框架,它提供了一系列的開(kāi)箱即用的功能和組件,這篇文章主要給大家介紹了關(guān)于spring?boot中spring框架的版本升級(jí)的相關(guān)資料,需要的朋友可以參考下2023-10-10springMVC如何將controller中Model數(shù)據(jù)傳遞到j(luò)sp頁(yè)面
本篇文章主要介紹了springMVC如何將controller中Model數(shù)據(jù)傳遞到j(luò)sp頁(yè)面,具有一定的參考價(jià)值,有興趣的可以了解一下2017-07-07