深入解讀Quartz的原理

weixin_33686714發表於2018-05-28
 Quartz是一個大名鼎鼎的Java版開源定時排程器,功能強悍,使用方便。
 
一、核心概念
 
Quartz的原理不是很複雜,只要搞明白幾個概念,然後知道如何去啟動和關閉一個排程程式即可。
 
1、Job
表示一個工作,要執行的具體內容。此介面中只有一個方法
void execute(JobExecutionContext context)

2、JobDetail

JobDetail表示一個具體的可執行的排程程式,Job是這個可執行程排程程式所要執行的內容,另外JobDetail還包含了這個任務排程的方案和策略。
 
3、Trigger代表一個排程引數的配置,什麼時候去調。
 
4、Scheduler代表一個排程容器,一個排程容器中可以註冊多個JobDetail和Trigger。當Trigger與JobDetail組合,就可以被Scheduler容器排程了。
 
 
二、一個最簡單入門例項
 
import org.quartz.*; 
import org.quartz.impl.StdSchedulerFactory;  
import java.util.Date;  
/** 
* quartz定時器測試 
* 
* @author leizhimin 2009-7-23 8:49:01 
*/ 
public class MyJob implements Job { 
        public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException { 
                System.out.println(new Date() + ": doing something..."); 
        } 
} 
class Test { 
        public static void main(String[] args) { 
                //1、建立JobDetial物件 
                JobDetail jobDetail = new JobDetail(); 
                //設定工作項 
                jobDetail.setJobClass(MyJob.class); 
                jobDetail.setName("MyJob_1"); 
                jobDetail.setGroup("JobGroup_1"); 
 
                //2、建立Trigger物件 
                SimpleTrigger strigger = new SimpleTrigger(); 
                strigger.setName("Trigger_1"); 
                strigger.setGroup("Trigger_Group_1"); 
                strigger.setStartTime(new Date()); 
                //設定重複停止時間,並銷燬該Trigger物件 
                java.util.Calendar c = java.util.Calendar.getInstance(); 
                c.setTimeInMillis(System.currentTimeMillis() + 1000 * 1L); 
                strigger.setEndTime(c.getTime()); 
                strigger.setFireInstanceId("Trigger_1_id_001"); 
                //設定重複間隔時間 
                strigger.setRepeatInterval(1000 * 1L); 
                //設定重複執行次數 
                strigger.setRepeatCount(3); 
 
                //3、建立Scheduler物件,並配置JobDetail和Trigger物件 
                SchedulerFactory sf = new StdSchedulerFactory(); 
                Scheduler scheduler = null; 
                try { 
                        scheduler = sf.getScheduler(); 
                        scheduler.scheduleJob(jobDetail, strigger); 
                        //4、並執行啟動、關閉等操作 
                        scheduler.start(); 
 
                } catch (SchedulerException e) { 
                        e.printStackTrace(); 
                } 
//                try { 
//                        //關閉排程器 
//                        scheduler.shutdown(true); 
//                } catch (SchedulerException e) { 
//                        e.printStackTrace(); 
//                } 
        } 
}

 執行結果:

 
當把結束時間改為:
             
   //設定重複停止時間,並銷燬該Trigger物件 
                java.util.Calendar c = java.util.Calendar.getInstance(); 
                c.setTimeInMillis(System.currentTimeMillis() + 1000 * 1L); 
                strigger.setEndTime(c.getTime());

執行結果:

 
當新增一條關閉排程器的語句:
                 
       //4、並執行啟動、關閉等操作 
                        scheduler.start(); 
                        scheduler.shutdown(true); 

程式執行結果:

Thu Jul 23 10:11:50 CST 2009: doing something... 

Process finished with exit code 0

僅僅執行了一次,這一次能執行完,原因是設定了scheduler.shutdown(true);true表示等待本次任務執行完成後停止。
 
從這裡也可以看出,scheduler是個容器,scheduler控制jobDetail的執行,控制的策略是通過trigger。
 
當scheduler容器啟動後,jobDetail才能根據關聯的trigger策略去執行。當scheduler容器關閉後,所有的jobDetail都停止執行。
 
三、透過例項看原理
 
通過研讀Quartz的原始碼,和本例項,終於悟出了Quartz的工作原理。
 
1、scheduler是一個計劃排程器容器(總部),容器裡面可以盛放眾多的JobDetail和trigger,當容器啟動後,裡面的每個JobDetail都會根據trigger按部就班自動去執行。
 
2、JobDetail是一個可執行的工作,它本身可能是有狀態的。
 
3、Trigger代表一個排程引數的配置,什麼時候去調。
 
4、當JobDetail和Trigger在scheduler容器上註冊後,形成了裝配好的作業(JobDetail和Trigger所組成的一對兒),就可以伴隨容器啟動而排程執行了。
 
5、scheduler是個容器,容器中有一個執行緒池,用來並行排程執行每個作業,這樣可以提高容器效率。
 
6、將上述的結構用一個圖來表示,如下:
 
 
 
四、總結
 
1、搞清楚了上Quartz容器執行作業的的原理和過程,以及作業形成的方式,作業註冊到容器的方法。就認識明白了Quartz的核心原理。
 
2、Quartz雖然很龐大,但是一切都圍繞這個核心轉,為了配置強大時間排程策略,可以研究專門的CronTrigger。要想靈活配置作業和容器屬性,可以通過Quartz的properties檔案或者XML來實現。
 
3、要想排程更多的持久化、結構化作業,可以通過資料庫讀取作業,然後放到容器中執行。
 
4、所有的一切都圍繞這個核心原理轉,搞明白這個了,再去研究更高階用法就容易多了。
 
5、Quartz與Spring的整合也非常簡單,Spring提供一組Bean來支援:MethodInvokingJobDetailFactoryBean、SimpleTriggerBean、SchedulerFactoryBean,看看裡面需要注入什麼屬性即可明白了。Spring會在Spring容器啟動時候,啟動Quartz容器。
 
6、Quartz容器的關閉方式也很簡單,如果是Spring整合,則有兩種方法,一種是關閉Spring容器,一種是獲取到SchedulerFactoryBean例項,然後呼叫一個shutdown就搞定了。如果是Quartz獨立使用,則直接呼叫scheduler.shutdown(true);
 
7、Quartz的JobDetail、Trigger都可以在執行時重新設定,並且在下次呼叫時候起作用。這就為動態作業的實現提供了依據。你可以將排程時間策略存放到資料庫,然後通過資料庫資料來設定Trigger,這樣就能產生動態的排程。

相關文章