Spring&Quartz集成自定義注釋

我們知道Spring支持與Quartz框架集成。 但是到目前為止,Spring僅支持靜態XML聲明方法。

如果想了解如何將Spring與Quartz集成,可以參考Spring + Quartz + JavaMail集成教程 。

作為寵物項目要求的一部分,我必須動態安排工作,并且想到了以下兩個選項:

1.使用注釋提供作業元數據
2.從數據庫加載作業元數據

現在,我想到了繼續使用基于注釋的方法,并且也希望將其與Spring集成。 這是我的方法。

1.創建一個自定義注解QuartzJob

package com.sivalabs.springsamples.jobscheduler;import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;import org.springframework.stereotype.Component;@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
@Scope("prototype")
public @interface QuartzJob 
{String name();String group() default "DEFAULT_GROUP";String cronExp();
}

2.創建一個ApplicationListener來掃描所有Job實施類,并使用Quartz Scheduler調度它們。

package com.sivalabs.springsamples.jobscheduler;import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Set;import org.quartz.Job;
import org.quartz.JobDetail;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.core.annotation.AnnotationUtils;
import org.springframework.scheduling.quartz.CronTriggerBean;
import org.springframework.scheduling.quartz.JobDetailBean;public class QuartJobSchedulingListener implements ApplicationListener<ContextRefreshedEvent>
{ @Autowiredprivate Scheduler scheduler;@Overridepublic void onApplicationEvent(ContextRefreshedEvent event){try {ApplicationContext applicationContext = event.getApplicationContext();List<CronTriggerBean> cronTriggerBeans = this.loadCronTriggerBeans(applicationContext);this.scheduleJobs(cronTriggerBeans);} catch (Exception e) {e.printStackTrace();}}private List<CronTriggerBean> loadCronTriggerBeans(ApplicationContext applicationContext){Map<String, Object> quartzJobBeans = applicationContext.getBeansWithAnnotation(QuartzJob.class);Set<String> beanNames = quartzJobBeans.keySet();List<CronTriggerBean> cronTriggerBeans = new ArrayList<CronTriggerBean>();for (String beanName : beanNames) {CronTriggerBean cronTriggerBean = null;Object object = quartzJobBeans.get(beanName);System.out.println(object);try {cronTriggerBean = this.buildCronTriggerBean(object);} catch (Exception e) {e.printStackTrace();}if(cronTriggerBean != null){cronTriggerBeans.add(cronTriggerBean);}}return cronTriggerBeans;}public CronTriggerBean buildCronTriggerBean(Object job) throws Exception{CronTriggerBean cronTriggerBean = null;QuartzJob quartzJobAnnotation = AnnotationUtils.findAnnotation(job.getClass(), QuartzJob.class);if(Job.class.isAssignableFrom(job.getClass())){System.out.println("It is a Quartz Job");cronTriggerBean = new CronTriggerBean();cronTriggerBean.setCronExpression(quartzJobAnnotation.cronExp());    cronTriggerBean.setName(quartzJobAnnotation.name()+"_trigger");//cronTriggerBean.setGroup(quartzJobAnnotation.group());JobDetailBean jobDetail = new JobDetailBean();jobDetail.setName(quartzJobAnnotation.name());//jobDetail.setGroup(quartzJobAnnotation.group());jobDetail.setJobClass(job.getClass());cronTriggerBean.setJobDetail(jobDetail);   }else{throw new RuntimeException(job.getClass()+" doesn't implemented "+Job.class);}return cronTriggerBean;}protected void scheduleJobs(List<CronTriggerBean> cronTriggerBeans){for (CronTriggerBean cronTriggerBean : cronTriggerBeans) {JobDetail jobDetail = cronTriggerBean.getJobDetail();try {scheduler.scheduleJob(jobDetail, cronTriggerBean);} catch (SchedulerException e) {e.printStackTrace();}   }}
}

3.創建一個自定義的JobFactory,以將Spring bean用作Job實現對象。

package com.sivalabs.springsamples.jobscheduler;import org.quartz.Job;
import org.quartz.spi.TriggerFiredBundle;
import org.springframework.beans.BeanWrapper;
import org.springframework.beans.MutablePropertyValues;
import org.springframework.beans.PropertyAccessorFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.scheduling.quartz.SpringBeanJobFactory;public class SpringQuartzJobFactory extends SpringBeanJobFactory
{@Autowiredprivate ApplicationContext ctx;@Overrideprotected Object createJobInstance(TriggerFiredBundle bundle) throws Exception {@SuppressWarnings("unchecked")Job job = ctx.getBean(bundle.getJobDetail().getJobClass());BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(job);MutablePropertyValues pvs = new MutablePropertyValues();pvs.addPropertyValues(bundle.getJobDetail().getJobDataMap());pvs.addPropertyValues(bundle.getTrigger().getJobDataMap());bw.setPropertyValues(pvs, true);return job;} 
}

4.創建Job實施類,并使用@QuartzJob對其進行批注

package com.sivalabs.springsamples.jobscheduler;import java.util.Date;import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.quartz.QuartzJobBean;@QuartzJob(name="HelloJob", cronExp="0/5 * * * * ?")
public class HelloJob extends QuartzJobBean
{  @Overrideprotected void executeInternal(JobExecutionContext context)throws JobExecutionException{System.out.println("Hello Job is running @ "+new Date());System.out.println(this.hashCode());  }
}

5.在applicationContext.xml中配置SchedulerFactoryBean和QuartJobSchedulingListener

<beans><context:annotation-config></context:annotation-config><context:component-scan base-package="com.sivalabs"></context:component-scan><bean class="com.sivalabs.springsamples.jobscheduler.QuartJobSchedulingListener"></bean><bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean"><property name="jobFactory"><bean class="com.sivalabs.springsamples.jobscheduler.SpringQuartzJobFactory"></bean></property></bean></beans>

6.使用測試客戶端啟動上下文

package com.sivalabs.springsamples;import org.quartz.Job;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;import com.sivalabs.springsamples.jobscheduler.HowAreYouJob;
import com.sivalabs.springsamples.jobscheduler.InvalidJob;public class TestClient
{public static void main(String[] args){ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");System.out.println(context);  }}

參考: JCG合作伙伴 Siva在“我的技術實驗”博客上的 使用自定義批注進行Spring和Quartz集成 。

相關文章 :
  • Spring,Quartz和JavaMail集成教程
  • 在運行時交換出Spring Bean配置
  • Spring MVC3 Hibernate CRUD示例應用程序
  • 使用Spring將POJO公開為JMX MBean
  • Java教程和Android教程列表

翻譯自: https://www.javacodegeeks.com/2011/10/spring-quartz-integration-with-custom.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/374263.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/374263.shtml
英文地址,請注明出處:http://en.pswp.cn/news/374263.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Android 服務

Android服務是Android應用程序的一類可以異步運行的組件 要創建自己的服務類&#xff0c;需要派生Service類&#xff0c;并至少用自定義代碼實現onCreate()、onStart()、onDestory()這幾個方法。此外還必須在 AndroidManifest.XML文件中用<service>標簽表明你的服務 <…

單片機/嵌入式軟件架構分層思想

以STM32裸機開發為例。 軟件分層應用層驅動層硬件層固件層 ①最底層為固件層&#xff0c;Firmware 這一層通常是官方給的庫&#xff0c;庫函數對寄存器進行操作&#xff0c;例如&#xff1a; /*** brief Transmits a Data through the SPIx/I2Sx peripheral.* param SPIx: …

玩! 框架:為什么我會愛上它

前一段時間&#xff0c;我是房地美&#xff0c;房地美&#xff0c;Foreclosure.com和HUD等公司在房地產市場上進行一些大型部署的技術負責人。 我們運行的是您可能熟悉的傳統企業Java堆棧-Spring &#xff0c; Hibernate &#xff0c;Solr等。花了幾年時間&#xff0c;但我們建…

關于在移動網頁中圖片自適應大小的寫法

一般在移動網頁時&#xff0c;圖片屬性寫成如下就可以達到自適應大小 <style type"text/css"> .nameg{background: rgba(000,000,000,0.6);} .nameg div{float: left;} .nameg .a1{width: 10%;background:#000000;} .nameg .a1 img{width: 100%;height: 10…

python2 print_Python2和Python3中print的不同點

在Python2和Python3中都提供print()方法來打印信息,但兩個版本間的print稍微有差異 主要體現在以下幾個方面&#xff1a; 1.python3中print是一個內置函數&#xff0c;有多個參數&#xff0c;而python2中print是一個語法結構&#xff1b; 2.Python2打印時可以不加括號&#xff…

java 與 c#的 中 字符串比較“==”與“equals”的差異

.net中&#xff0c;其字符串特有的駐留機制&#xff0c;保證了在同一進程中&#xff0c;相同字符序列的字符串&#xff0c;只有一個實例&#xff0c;這樣能避免相同內容的字符串重復實例化&#xff0c;以減少性能開銷。 先來回顧一下c#中的代碼&#xff1a; public static void…

visual studio 2019 未能在命名空間“System.IO.Ports”中找到類型名“SerialPort”

在vs2019以前的版本&#xff0c;只要using System.IO.Ports就可以用SerialPort。 這里需要自己手動添加相關引用。 工具–>Nuget包管理器&#xff08;N&#xff09;–>管理解決方案的Nuget程序包&#xff08;N&#xff09; –>瀏覽&#xff0c;左邊搜索SerialPort 右…

單一登錄云:SAML和OpenId

當訪問不同組織擁有的不同應用程序時&#xff0c;每次從一個應用程序轉到另一個應用程序時都必須進行身份驗證。 不僅浪費時間&#xff0c;而且您還必須記住多個經常丟失的密碼。 單一登錄是一次認證的能力&#xff0c;并且能夠使用已認證的身份在應用程序之間無縫切換。 在In…

python開發環境功能介紹_第一模塊 第3章 Python介紹與環境配置

python入門(全為重點) 1. 編程語言介紹 編程語言分類、總結 2. python介紹 3. 解釋器多版本共存 4. 運行python程序的兩種方式 5. 一個python程序運行的三個步驟&#xff08;******&#xff09; 6. 注釋 7. IED集成開發環境 3.1 編程語言分類之低級語言 這里的高級/低級指的是離…

如何判斷微信內置瀏覽器(JS PHP)

進行微信公眾賬號開發的時候&#xff0c;其中很大一塊是微站點的開發&#xff0c;我們需要知道當前的瀏覽器是微信內置的瀏覽器&#xff0c;那么如何判斷呢&#xff1f; 微信內置瀏覽器的 User Agent Mozilla/5.0 (iPhone; CPU iPhone OS 6_1_3 like Mac OS X) AppleWebKit/536…

用WPF做關于MEF 簡單學習記錄

寫在前面&#xff1a;下面學習所得多是從自http://www.cnblogs.com/comsokey/p/MEF1.html和http://www.cnblogs.com/yunfeifei/p/3922668.html兩位大神的文章里學到的&#xff0c;特別鳴謝&#xff01;整理下是更大一方面是對自己知識的梳理&#xff0c;用詞用句不夠準確&#…

Log4j,Stat4j,SMTPAppender集成–匯總錯誤日志以發送太多電子郵件

我們的開發團隊希望在生產系統出現問題時盡快得到通知&#xff0c;這是一個每天為成千上萬的客戶提供服務的關鍵Java Web應用程序。 這個想法是讓它在出現太多錯誤時向我們發送電子郵件&#xff0c;這些錯誤通常表明數據庫&#xff0c;外部Web服務存在問題&#xff0c;或者應用…

C排序算法:(一)直接排序

最簡單的排序方法。 如果從大到小排序&#xff0c;那么從[0]元素開始&#xff0c;和后面的元素進行對比&#xff0c;如果后面元素大&#xff0c;則和[0]元素交換。 核心思想&#xff1a;選定基準元素&#xff0c;和其他元素對比。 #include <stdio.h>unsigned char buff…

數據庫抽象類PDOStatement對象使用

1.預處理語句中使用占位符,分為 索引方式 和 關聯方式 a.索引方式&#xff1a; $pdonew PDO($dns,$username,$password,$options); $sqlselect * from tests where username?,password?; $stmt$pdo->prepare($sql); $stmt->execute(array($username,$password)); b.關聯…

c語言 鏈表_C語言編程第22講——單向有序鏈表的C語言實現

1、單向有序鏈表的含義單向有序鏈表可以解析為四個名詞&#xff1a;表&#xff1a;一組元素&#xff1b;鏈表&#xff1a;表中的元素不是從前往后一個挨著一個&#xff0c;而是通過一個元素才能找到另一個元素&#xff1b;單向&#xff1a;表中的元素只能從前往后訪問&#xff…

Spring 3.1和Hibernate的持久層

1.概述 本文將展示如何使用Spring和Hibernate實現DAO 。 有關核心的Hibernate配置&#xff0c;請參閱有關帶??有Spring的Hibernate 3和Hibernate 4的文章。 2.沒有更多的春天模板 從Spring 3.0和Hibernate 3.0.1開始&#xff0c; 不再需要Spring HibernateTemplate來管理Hib…

C排序算法:(二)冒泡排序

冒泡排序就是從左至右比較相鄰的兩個數值大小&#xff0c;如果右側的數值較小&#xff0c;則交換兩個數值的位置&#xff0c;較大的數值就會像泡泡一樣一路向右漂浮。 #include <stdio.h>//small to big void Bubble_Sort(unsigned char *input_data, unsigned int inpu…

C語言之猜數游戲

#include<stdio.h>#include<stdlib.h>#include<time.h>int main(){ srand(time(0)); int count0,thought; int numrand()%1001; printf("%d\n",num); do { printf("請猜這個數在1-100之間\n"); scanf("%d",&thought); cou…

pythonturtle畫點的指令_簡述python的turtle繪畫命令及解釋

一 基礎認識 turtle庫是python的標準庫之一&#xff0c;它是一個直觀有趣的圖形繪制數據庫&#xff0c;turtle(海龜&#xff09;圖形繪制的概念誕生1969年。它的應用十分廣&#xff0c;而且使用簡單&#xff0c;只要在編寫python程序時寫上import turtle即可。 1.繪圖窗口設置命…

算法—振興中華(C語言版)

/* 標題: 振興中華小明參加了學校的趣味運動會&#xff0c;其中的一個項目是&#xff1a;跳格子。地上畫著一些格子&#xff0c;每個格子里寫一個字&#xff0c;如下所示&#xff1a;從我做起振我做起振興做起振興中起振興中華比賽時&#xff0c;先站在左上角的寫著“從”字的格…