Quartz和Spring Task定時任務的簡單應用和比較

看了兩個項目,一個用的是Quartz寫的定時器,一個是使用spring的task寫的,網上看了2篇文章,寫的比較清楚,這裏做一下留存
鏈接一、菠蘿大象:http://www.blogjava.net/bolo/archive/2015/03/12/423408.html

連接二:http://huangrongyou.iteye.com/blog/1762869

一、Quartz

  • 引入quartz的jar包。
  • 配置文件中定義org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean,並指定它的targetObject屬性爲Job任務類,targetMethod屬性爲任務方法就可以了。

    <bean id="job" class=" xx.xx.xx.Job" />
    <bean id="cronTask" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
        <property name="targetObject" ref="job" />
        <property name="targetMethod" value="runWork" />
        <!-- false表示job不會併發執行,默認爲true-->
        <property name="concurrent" value="false" />
    </bean>
    targetObject屬性指定的任務類,有多種方式實現。
    1、可以用@Component註解在類上面標註,這樣就不用定義<bean id="job" ... />這些東西了。
    2、可以按上面的寫法來配置。
    3、直接使用下面的寫法。
    <property name="targetObject">
        <bean class="xx.xx.xx.Job" />
    </property>
    接下來配置觸發器
    <bean id="doWork" class="org.springframework.scheduling.quartz.CronTriggerBean">
        <property name="jobDetail" ref="cronTask" />
        <!—每天凌晨01分執行-->
        <property name="cronExpression" value="0 01 00 * * ?" />
    </bean>
    最後配置調度工廠
    <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
        <property name="triggers">
            <list>
                <ref local="doWork"/>
            </list>
        </property>
    </bean>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 到此,整個配置就完成了。
  • 時間參考:
    “0/10 * * * * ?” 每10秒觸發
    “0 0 12 * * ?” 每天中午12點觸發
    “0 15 10 ? * *” 每天上午10:15觸發
    “0 15 10 * * ?” 每天上午10:15觸發
    “0 15 10 * * ? *” 每天上午10:15觸發
    “0 15 10 * * ? 2005” 2005年的每天上午10:15觸發
    “0 * 14 * * ?” 在每天下午2點到下午2:59期間的每1分鐘觸發
    “0 0/5 14 * * ?” 在每天下午2點到下午2:55期間的每5分鐘觸發
    “0 0/5 14,18 * * ?” 在每天下午2點到2:55期間和下午6點到6:55期間的每5分鐘觸發
    “0 0-5 14 * * ?” 在每天下午2點到下午2:05期間的每1分鐘觸發
    “0 10,44 14 ? 3 WED” 每年三月的星期三的下午2:10和2:44觸發
    “0 15 10 ? * MON-FRI” 週一至週五的上午10:15觸發
    “0 15 10 15 * ?” 每月15日上午10:15觸發
    “0 15 10 L * ?” 每月最後一日的上午10:15觸發
    “0 15 10 ? * 6L” 每月的最後一個星期五上午10:15觸發
    “0 15 10 ? * 6L 2002-2005” 2002年至2005年的每月的最後一個星期五上午10:15觸發
    “0 15 10 ? * 6#3” 每月的第三個星期五上午10:15觸發
    每隔5秒執行一次:/5 * * * ?
    每隔1分鐘執行一次:0 /1 * * ?
    每天23點執行一次:0 0 23 * * ?
    每天凌晨1點執行一次:0 0 1 * * ?
    每月1號凌晨1點執行一次:0 0 1 1 * ?
    每月最後一天23點執行一次:0 0 23 L * ?
    每週星期天凌晨1點實行一次:0 0 1 ? * L
    在26分、29分、33分執行一次:0 26,29,33 * * * ?
    每天的0點、13點、18點、21點都執行一次:0 0 0,13,18,21 * * ?

Spring Task

  • Spring從3.0開始增加了自己的任務調度器,它是通過擴展java.util.concurrent包下面的類來實現的,它也使用Cron表達式。
  • 使用spring task非常簡單,首先增加命名空間schema

    <beans xmlns="http://www.springframework.org/schema/beans" 
        ......
        xmlns:task="http://www.springframework.org/schema/task"
            xsi:schemaLocation="
            ......
        http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd">
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 然後給定時任務類添加@Component註解,給任務方法添加@Scheduled(cron = “0/5 * * * * ?”)註解,並讓Spring掃描到該類。
    然後加上這個配置,讓Spring識別@Scheduled註解(org.springframework.scheduling.annotation.Scheduled)。
  • 如果還想擴展一下,改成下面這樣:

    <task:executor id="executor" pool-size="5" />
    <task:scheduler id="scheduler" pool-size="5" />
    <task:annotation-driven executor="executor" scheduler="scheduler" />
  • 1
  • 2
  • 3
  • 4
  • 如果定時任務很多,可以配置executor線程池,這裏executor的含義和java.util.concurrent.Executor是一樣的,pool-size的大小官方推薦爲5~10。scheduler的pool-size是ScheduledExecutorService線程池,默認爲1。假如我設置了8個任務,每個任務都是每5秒鐘執行一次,把下面的代碼再複製7份再改一改,看看打印結果。

    @Scheduled(cron = "0/5 * * * * ?")
    public void work1(){
        System.out.println(Thread.currentThread().getName()+" "+"work1: 每5秒執行一次");
    }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

  • 定時任務執行了3次,我們可以看到,線程名稱都是以scheduler爲前綴,這是因爲我們已經在這段配置裏定義了id爲scheduler的結果,它就是用來作爲任務線程的前綴,再交給executor線程池進行。3次任務執行,因爲我們設定的任務調度線程池大小爲5,所以,只有5個實例來處理這8個任務,從結果可以看出來,不是每次都會用上全部的5個實例。如果你係統中的定時任務過多,這個pool-size的大小就應該調大一點,方便之前定義的executor線程池來執行。

比較

轉載的博主寫了很多測試類,這裏就不貼了,客官可以轉到鏈接二查看,這裏只寫一下區別

  • 精確度和功能
    • Quartz可以通過cron表達式精確到特定時間執行,而TimerTask不能。
    • Quartz擁有TimerTask所有的功能,而TimerTask則沒有。

  • 任務類的數量
    • Quartz每次執行都創建一個新的任務類對象。
    • TimerTask則每次使用同一個任務類對象。

  • 對異常的處理

    • Quartz的某次執行任務過程中拋出異常,不影響下一次任務的執行,當下一次執行時間到來時,定時器會再次執行任務。
    • TimerTask不同,一旦某個任務在執行過程中拋出異常,則整個定時器生命週期就結束,以後永遠不會再執行定時器任務。
  • 總結:還是第三方考慮的周全,東西比較多,加上Quartz配置簡單,maven也就是多加一個jar包,所以一般情況下還是使用Quartz了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章