在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

SpringBoot如何實現動態增刪啟停定時任務

Android編程精選 ? 來源:簡書 ? 作者:jessehua ? 2021-09-24 09:49 ? 次閱讀
在spring boot項目中,可以通過@EnableScheduling注解和@Scheduled注解實現定時任務,也可以通過SchedulingConfigurer接口來實現定時任務。但是這兩種方式不能動態添加、刪除、啟動、停止任務。

要實現動態增刪啟停定時任務功能,比較廣泛的做法是集成Quartz框架。但是本人的開發原則是:在滿足項目需求的情況下,盡量少的依賴其它框架,避免項目過于臃腫和復雜。

查看spring-context這個jar包中org.springframework.scheduling.ScheduledTaskRegistrar這個類的源代碼,發現可以通過改造這個類就能實現動態增刪啟停定時任務功能。

e8f20370-1085-11ec-8fb8-12bb97331649.jpg定時任務列表頁e90122f6-1085-11ec-8fb8-12bb97331649.jpg定時任務執行日志

添加執行定時任務的線程池配置類

@Configuration
publicclassSchedulingConfig{
@Bean
publicTaskSchedulertaskScheduler(){
ThreadPoolTaskSchedulertaskScheduler=newThreadPoolTaskScheduler();
//定時任務執行線程池核心線程數
taskScheduler.setPoolSize(4);
taskScheduler.setRemoveOnCancelPolicy(true);
taskScheduler.setThreadNamePrefix("TaskSchedulerThreadPool-");
returntaskScheduler;
}
}

添加ScheduledFuture的包裝類。ScheduledFuture是ScheduledExecutorService定時任務線程池的執行結果。

publicfinalclassScheduledTask{

volatileScheduledFuturefuture;

/**
*取消定時任務
*/
publicvoidcancel(){
ScheduledFuturefuture=this.future;
if(future!=null){
future.cancel(true);
}
}
}

添加Runnable接口實現類,被定時任務線程池調用,用來執行指定bean里面的方法。

publicclassSchedulingRunnableimplementsRunnable{

privatestaticfinalLoggerlogger=LoggerFactory.getLogger(SchedulingRunnable.class);

privateStringbeanName;

privateStringmethodName;

privateStringparams;

publicSchedulingRunnable(StringbeanName,StringmethodName){
this(beanName,methodName,null);
}

publicSchedulingRunnable(StringbeanName,StringmethodName,Stringparams){
this.beanName=beanName;
this.methodName=methodName;
this.params=params;
}

@Override
publicvoidrun(){
logger.info("定時任務開始執行- bean:{},方法:{},參數:{}",beanName,methodName,params);
longstartTime=System.currentTimeMillis();

try{
Objecttarget=SpringContextUtils.getBean(beanName);

Methodmethod=null;
if(StringUtils.isNotEmpty(params)){
method=target.getClass().getDeclaredMethod(methodName,String.class);
}else{
method=target.getClass().getDeclaredMethod(methodName);
}

ReflectionUtils.makeAccessible(method);
if(StringUtils.isNotEmpty(params)){
method.invoke(target,params);
}else{
method.invoke(target);
}
}catch(Exceptionex){
logger.error(String.format("定時任務執行異常- bean:%s,方法:%s,參數:%s ",beanName,methodName,params),ex);
}

longtimes=System.currentTimeMillis()-startTime;
logger.info("定時任務執行結束- bean:{},方法:{},參數:{},耗時:{}毫秒",beanName,methodName,params,times);
}

@Override
publicbooleanequals(Objecto){
if(this==o)returntrue;
if(o==null||getClass()!=o.getClass())returnfalse;
SchedulingRunnablethat=(SchedulingRunnable)o;
if(params==null){
returnbeanName.equals(that.beanName)&&
methodName.equals(that.methodName)&&
that.params==null;
}

returnbeanName.equals(that.beanName)&&
methodName.equals(that.methodName)&&
params.equals(that.params);
}

@Override
publicinthashCode(){
if(params==null){
returnObjects.hash(beanName,methodName);
}

returnObjects.hash(beanName,methodName,params);
}
}

添加定時任務注冊類,用來增加、刪除定時任務。

@Component
publicclassCronTaskRegistrarimplementsDisposableBean{

privatefinalMapscheduledTasks=newConcurrentHashMap<>(16);

@Autowired
privateTaskSchedulertaskScheduler;

publicTaskSchedulergetScheduler(){
returnthis.taskScheduler;
}

publicvoidaddCronTask(Runnabletask,StringcronExpression){
addCronTask(newCronTask(task,cronExpression));
}

publicvoidaddCronTask(CronTaskcronTask){
if(cronTask!=null){
Runnabletask=cronTask.getRunnable();
if(this.scheduledTasks.containsKey(task)){
removeCronTask(task);
}

this.scheduledTasks.put(task,scheduleCronTask(cronTask));
}
}

publicvoidremoveCronTask(Runnabletask){
ScheduledTaskscheduledTask=this.scheduledTasks.remove(task);
if(scheduledTask!=null)
scheduledTask.cancel();
}

publicScheduledTaskscheduleCronTask(CronTaskcronTask){
ScheduledTaskscheduledTask=newScheduledTask();
scheduledTask.future=this.taskScheduler.schedule(cronTask.getRunnable(),cronTask.getTrigger());

returnscheduledTask;
}


@Override
publicvoiddestroy(){
for(ScheduledTasktask:this.scheduledTasks.values()){
task.cancel();
}

this.scheduledTasks.clear();
}
}

添加定時任務示例類

@Component("demoTask")
publicclassDemoTask{
publicvoidtaskWithParams(Stringparams){
System.out.println("執行有參示例任務:"+params);
}

publicvoidtaskNoParams(){
System.out.println("執行無參示例任務");
}
}

定時任務數據庫表設計

添加定時任務實體類

publicclassSysJobPO{
/**
*任務ID
*/
privateIntegerjobId;
/**
*bean名稱
*/
privateStringbeanName;
/**
*方法名稱
*/
privateStringmethodName;
/**
*方法參數
*/
privateStringmethodParams;
/**
*cron表達式
*/
privateStringcronExpression;
/**
*狀態(1正常0暫停)
*/
privateIntegerjobStatus;
/**
*備注
*/
privateStringremark;
/**
*創建時間
*/
privateDatecreateTime;
/**
*更新時間
*/
privateDateupdateTime;

publicIntegergetJobId(){
returnjobId;
}

publicvoidsetJobId(IntegerjobId){
this.jobId=jobId;
}

publicStringgetBeanName(){
returnbeanName;
}

publicvoidsetBeanName(StringbeanName){
this.beanName=beanName;
}

publicStringgetMethodName(){
returnmethodName;
}

publicvoidsetMethodName(StringmethodName){
this.methodName=methodName;
}

publicStringgetMethodParams(){
returnmethodParams;
}

publicvoidsetMethodParams(StringmethodParams){
this.methodParams=methodParams;
}

publicStringgetCronExpression(){
returncronExpression;
}

publicvoidsetCronExpression(StringcronExpression){
this.cronExpression=cronExpression;
}

publicIntegergetJobStatus(){
returnjobStatus;
}

publicvoidsetJobStatus(IntegerjobStatus){
this.jobStatus=jobStatus;
}

publicStringgetRemark(){
returnremark;
}

publicvoidsetRemark(Stringremark){
this.remark=remark;
}

publicDategetCreateTime(){
returncreateTime;
}

publicvoidsetCreateTime(DatecreateTime){
this.createTime=createTime;
}

publicDategetUpdateTime(){
returnupdateTime;
}

publicvoidsetUpdateTime(DateupdateTime){
this.updateTime=updateTime;
}

}
booleansuccess=sysJobRepository.addSysJob(sysJob);
if(!success)
returnOperationResUtils.fail("新增失敗");
else{
if(sysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())){
SchedulingRunnabletask=newSchedulingRunnable(sysJob.getBeanName(),sysJob.getMethodName(),sysJob.getMethodParams());
cronTaskRegistrar.addCronTask(task,sysJob.getCronExpression());
}
}

returnOperationResUtils.success();

修改定時任務,先移除原來的任務,再啟動新任務

booleansuccess=sysJobRepository.editSysJob(sysJob);
if(!success)
returnOperationResUtils.fail("編輯失敗");
else{
//先移除再添加
if(existedSysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())){
SchedulingRunnabletask=newSchedulingRunnable(existedSysJob.getBeanName(),existedSysJob.getMethodName(),existedSysJob.getMethodParams());
cronTaskRegistrar.removeCronTask(task);
}

if(sysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())){
SchedulingRunnabletask=newSchedulingRunnable(sysJob.getBeanName(),sysJob.getMethodName(),sysJob.getMethodParams());
cronTaskRegistrar.addCronTask(task,sysJob.getCronExpression());
}
}

returnOperationResUtils.success();

刪除定時任務

booleansuccess=sysJobRepository.deleteSysJobById(req.getJobId());
if(!success)
returnOperationResUtils.fail("刪除失敗");
else{
if(existedSysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())){
SchedulingRunnabletask=newSchedulingRunnable(existedSysJob.getBeanName(),existedSysJob.getMethodName(),existedSysJob.getMethodParams());
cronTaskRegistrar.removeCronTask(task);
}
}

returnOperationResUtils.success();

定時任務啟動/停止狀態切換

if(existedSysJob.getJobStatus().equals(SysJobStatus.NORMAL.ordinal())){
SchedulingRunnabletask=newSchedulingRunnable(existedSysJob.getBeanName(),existedSysJob.getMethodName(),existedSysJob.getMethodParams());
cronTaskRegistrar.addCronTask(task,existedSysJob.getCronExpression());
}else{
SchedulingRunnabletask=newSchedulingRunnable(existedSysJob.getBeanName(),existedSysJob.getMethodName(),existedSysJob.getMethodParams());
cronTaskRegistrar.removeCronTask(task);
}

添加實現了CommandLineRunner接口的SysJobRunner類,當spring boot項目啟動完成后,加載數據庫里狀態為正常的定時任務。

@Service
publicclassSysJobRunnerimplementsCommandLineRunner{

privatestaticfinalLoggerlogger=LoggerFactory.getLogger(SysJobRunner.class);

@Autowired
privateISysJobRepositorysysJobRepository;

@Autowired
privateCronTaskRegistrarcronTaskRegistrar;

@Override
publicvoidrun(String...args){
//初始加載數據庫里狀態為正常的定時任務
ListjobList=sysJobRepository.getSysJobListByStatus(SysJobStatus.NORMAL.ordinal());
if(CollectionUtils.isNotEmpty(jobList)){
for(SysJobPOjob:jobList){
SchedulingRunnabletask=newSchedulingRunnable(job.getBeanName(),job.getMethodName(),job.getMethodParams());
cronTaskRegistrar.addCronTask(task,job.getCronExpression());
}

logger.info("定時任務已加載完畢...");
}
}
}

工具類SpringContextUtils,用來從spring容器里獲取bean

@Component
publicclassSpringContextUtilsimplementsApplicationContextAware{

privatestaticApplicationContextapplicationContext;

@Override
publicvoidsetApplicationContext(ApplicationContextapplicationContext)
throwsBeansException{
SpringContextUtils.applicationContext=applicationContext;
}

publicstaticObjectgetBean(Stringname){
returnapplicationContext.getBean(name);
}

publicstaticTgetBean(ClassrequiredType){
returnapplicationContext.getBean(requiredType);
}

publicstaticTgetBean(Stringname,ClassrequiredType){
returnapplicationContext.getBean(name,requiredType);
}

publicstaticbooleancontainsBean(Stringname){
returnapplicationContext.containsBean(name);
}

publicstaticbooleanisSingleton(Stringname){
returnapplicationContext.isSingleton(name);
}

publicstaticClassgetType(Stringname){
returnapplicationContext.getType(name);
}
}

本文完,參考本文代碼可成功運行,親測!

(感謝閱讀,希望對你所有幫助)來源:www.jianshu.com/p/0f68936393fd
編輯:jq
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 源代碼
    +關注

    關注

    96

    文章

    2945

    瀏覽量

    66747
  • spring
    +關注

    關注

    0

    文章

    340

    瀏覽量

    14343
  • Boot
    +關注

    關注

    0

    文章

    149

    瀏覽量

    35837
  • SpringBoot
    +關注

    關注

    0

    文章

    173

    瀏覽量

    178

原文標題:告別硬編碼,SpringBoot實現動態增刪啟停定時任務

文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Linux計劃任務介紹

    1.計劃任務定時任務)基本概述 1.什么是crond crond就是計劃任務,類似于我們平時生活中的鬧鐘。定點執行。 2.為什么要使用crond?crond主要是做一些周期性的任務
    的頭像 發表于 11-24 15:49 ?287次閱讀

    定時器技術:Air780E如何革新定時任務管理?

    今天講的是關于Air780E如何革新定時任務管理的內容,希望大家有所收獲。
    的頭像 發表于 11-07 13:50 ?232次閱讀
    <b class='flag-5'>定時</b>器技術:Air780E如何革新<b class='flag-5'>定時任務</b>管理?

    mysql定時備份任務

    在生產環境上,為了避免數據的丟失,通常情況下都會定時的對數據庫進行備份。而Linux的crontab指令則可以幫助我們實現對數據庫定時進行備份。首先我們來簡單了解crontab指令,如果你會了請跳到下一個內容mysql備份。
    的頭像 發表于 10-31 10:07 ?163次閱讀

    變頻器外接按鈕如何接線

    以下是變頻器外接按鈕接線的一般步驟: 確定變頻器的型號和規格,查閱其技術手冊以獲取接線圖和接線說明。 準備所需的工具和材料,包括螺絲刀、剝線鉗、接線端子、接線盒等。 關閉電源并斷開變頻器與電源
    的頭像 發表于 08-25 10:49 ?1689次閱讀

    linux定時任務的用法總結

    習慣了使用 windows 的計劃任務,使用 linux 中的 crontab 管理定時任務時很不適應。
    的頭像 發表于 08-14 18:16 ?850次閱讀
    linux<b class='flag-5'>定時任務</b>的用法總結

    ESP8266如何實現時間小于3us的定時任務

    實現一個穩定的軟串口,現有的軟串口程序是通過中斷實現的,但中斷好像會被其他中斷打斷,導致數據丟失,定時器按文檔上的說法,只能大于50us,能不能實現時間小于3us的
    發表于 07-19 06:13

    定時器的工作方式介紹

    實現周期性事件的硬件模塊。它可以用于實現各種定時任務,如定時中斷、PWM(脈沖寬度調制)輸出、頻率測量等。定時器通常由一個計數器、一個時鐘
    的頭像 發表于 07-12 10:29 ?952次閱讀

    長持續時間定時器電路圖 時間定時器的工作原理和功能

    的處理,都離不開定時器的精確控制。時間定時器通常由硬件和軟件兩部分組成,硬件部分通過計時器芯片或計數器來實現時間的度量和計算,而軟件部分則是通過編程語言提供的函數或類庫來設置和處理定時任務
    的頭像 發表于 06-24 17:34 ?1894次閱讀
    長持續時間<b class='flag-5'>定時</b>器電路圖 時間<b class='flag-5'>定時</b>器的工作原理和功能

    PLC工程示例之步進電機

    電子發燒友網站提供《PLC工程示例之步進電機.rar》資料免費下載
    發表于 06-11 09:09 ?8次下載

    在物通博聯工業智能網關的本地配置界面(WEB)直接配置定時控制任務

    開關,可實現全年定時任務自動執行。在多個任務日期重疊時,可選執行高等級的還是并行執行。設有一個遠程和本地的控制點,默認為本地狀態,網關自己執行設置的任務,當用戶將改控制點切換為遠程時可
    的頭像 發表于 04-24 17:21 ?537次閱讀
    在物通博聯工業智能網關的本地配置界面(WEB)直接配置<b class='flag-5'>定時</b>控制<b class='flag-5'>任務</b>

    基于VC的三相異步電機系統的設計

    針對小功率三相異步電機的,當前國內外還較多地采用繼電器、接觸器等控制電器來實現自動控制。由于繼電接觸器控制系統是通過觸點的機械運動來通斷主、控回路,然而觸點因為電、機械和化學的原因易于磨損,并且在高沖擊、振蕩的工作環境下,觸
    發表于 02-15 17:21 ?516次閱讀
    基于VC的三相異步電機<b class='flag-5'>啟</b><b class='flag-5'>停</b>系統的設計

    使用TC21x的GPT實現1m計時器執行定時任務,怎么配置GTM和GPT?

    專家們好,我想使用TC21x的GPT實現1m計時器執行定時任務,不知道怎么配置GTM和GPT?
    發表于 02-06 06:47

    鴻蒙ArkUI開發-實現增刪Tab頁簽

    本文以瀏覽器中增加或刪除頁簽為例,實現Tabs中頁簽的增刪功能。
    的頭像 發表于 01-29 18:43 ?1587次閱讀
    鴻蒙ArkUI開發-<b class='flag-5'>實現</b><b class='flag-5'>增刪</b>Tab頁簽

    鴻蒙原生應用/元服務開發-長時任務

    概述 功能介紹 應用退至后臺后,對于在后臺需要長時間運行用戶可感知的任務,例如播放音樂、導航等。為防止應用進程被掛起,導致對應功能異常,可以申請長時任務,使應用在后臺長時間運行。申請長時任務后,系統
    發表于 01-09 10:52

    任務調度系統設計的核心邏輯

    Redis的讀寫性能極好,分布式鎖也比Quartz數據庫行級鎖更輕量級。當然Redis鎖也可以替換成Zookeeper鎖,也是同樣的機制。 在小型項目中,使用:定時任務框架(Quartz/Spring Schedule)和 分布式鎖(redis/zookeeper)有不錯的效果。
    的頭像 發表于 01-02 15:09 ?884次閱讀
    <b class='flag-5'>任務</b>調度系統設計的核心邏輯
    主站蜘蛛池模板: 亚洲免费网站在线观看| 四虎在线视频| 国产成人啪精品午夜在线观看 | 色爱区综合五月激情| 中文在线最新版天堂bt| 天天摸天天做天天爽水多| 日本黄色一级大片| 久久国产精品网| 91一区二区三区四区五区| 色综合小说天天综合网| 深爱五月激情| 最近2018中文字幕2019视频| 欧美成人 一区二区三区| 丁香六月婷婷七月激情| 色综合久久88色综合天天| 亚洲欧美色一区二区三区| 日本理论片www视频| 播色屋| 欧美午夜性春猛交bbb| 456亚洲人成影院在线观| 亚洲天天做日日做天天看2018| 全部在线播放免费毛片| www五月天| 国产欧美日韩haodiaose| 骚黄视频| 国产乱人视频在线看| 久久亚洲国产成人影院| 琪琪see色原在线20| 日韩美a一级毛片| 4虎.最新地址| 伊人黄色网| 亚洲第一成人影院| 久久久精品免费国产四虎| 天天色狠狠干| 又黄又爽又猛大片录像| 色婷婷亚洲精品综合影院| a亚洲天堂| 日一区二区三区| 国产婷婷色一区二区三区深爱网| 日本在线视| 美女扒开内裤无遮挡禁18|