解決Spring Batch框架job任務(wù)只跑一次的問題
Spring Batch job任務(wù)只跑一次
在一次實(shí)際使用spring batch的過程中,在定時(shí)任務(wù)中,第一次執(zhí)行Job沒有出現(xiàn)問題,然后再次執(zhí)行時(shí)不會(huì)執(zhí)行job任務(wù);
出現(xiàn)原因
針對(duì)這種異常需要明確Job Instance的概念,Job Instance 是由Job的名稱和執(zhí)行該job的參數(shù)組成的,當(dāng)執(zhí)行Job時(shí),會(huì)由于參數(shù)相同,會(huì)認(rèn)為是同一個(gè)Job實(shí)例,如果該Job已經(jīng)執(zhí)行過,就會(huì)報(bào)出異常。判斷Job是否執(zhí)行過的狀態(tài)是保存到Job Repository中的。
解決方法
為了使執(zhí)行參數(shù)不同,可以在參數(shù)中添加一個(gè)時(shí)間戳
JobParametersBuilder builder = new JobParametersBuilder(); builder.addDate("date", new Date());
job的啟動(dòng),停止,放棄
1、啟動(dòng)一個(gè)job
運(yùn)行一個(gè)批處理任務(wù)至少有兩點(diǎn)要求:一個(gè) JobLauncher 和一個(gè)用來運(yùn)行的 job 。它們都包含了相同或是不同的 context 。舉例來說,從命令行來啟動(dòng)job,會(huì)為每一個(gè)job初始化一個(gè)JVM,因此每個(gè)job會(huì)有一個(gè)自己的 JobLauncher;從web容器的HttpRequest來啟動(dòng)job,一般只是用一個(gè) JobLauncher 來異步啟動(dòng)job,http請求會(huì)調(diào)用這個(gè) JobLauncher 來啟動(dòng)它們需要的job。
通過web啟動(dòng)job的例子:
@Controller public class JobLauncherController { @Autowired JobLauncher jobLauncher; @Autowired Job job; @RequestMapping("/jobLauncher.html") public void handle() throws Exception{ jobLauncher.run(job, new JobParameters()); } }
2、停止一個(gè)job
關(guān)閉不是立即發(fā)生的,因?yàn)闆]有辦法將一個(gè)任務(wù)立刻強(qiáng)制停掉,尤其是當(dāng)任務(wù)進(jìn)行到開發(fā)人員自己的代碼段時(shí),框架在此刻是無能為力的,比如某個(gè)業(yè)務(wù)邏輯處理。而一旦控制權(quán)還給了框架,它會(huì)立刻設(shè)置當(dāng)前 StepExecution 為 BachStatus.STOPPED ,意為停止,然后保存,最后在完成前對(duì)JobExecution進(jìn)行相同的操作。
Set<Long> executions = jobOperator.getRunningExecutions("sampleJob"); jobOperator.stop(executions.iterator().next());
或者在配置文件中實(shí)現(xiàn):
它能夠讓job進(jìn)行短暫的停留,以便操作員有時(shí)間進(jìn)行其他的操作,stop元素必須配置restart屬性,當(dāng)該job重新啟動(dòng)的時(shí)候,需要手動(dòng)觸發(fā),執(zhí)行step2。
<step id="step1" parent="s1"> <stop on="COMPLETED" restart="step2"/> </step> <step id="step2" parent="s2"/>
3、放棄一個(gè)job
一個(gè)job的執(zhí)行過程當(dāng)執(zhí)行到FAILED狀態(tài)之后,如果它是可重啟的,它將會(huì)被重啟。如果任務(wù)的執(zhí)行過程狀態(tài)是ABANDONED,那么框架就不會(huì)重啟它。ABANDONED狀態(tài)也適用于執(zhí)行步驟,使得它們可以被跳過,即便是在一個(gè)可重啟的任務(wù)執(zhí)行之中:如果任務(wù)執(zhí)行過程中碰到在上一次執(zhí)行失敗后標(biāo)記為ABANDONED的步驟,將會(huì)跳過該步驟直接到下一步(這是由任務(wù)流定義和執(zhí)行步驟的退出碼決定的)。
如果當(dāng)前的系統(tǒng)進(jìn)程死掉了(“kill -9”或系統(tǒng)錯(cuò)誤),job自然也不會(huì)運(yùn)行,但JobRepository是無法偵測到這個(gè)錯(cuò)誤的,因?yàn)檫M(jìn)程死掉之前沒有對(duì)它進(jìn)行任何通知。你必須手動(dòng)的告訴它,你知道任務(wù)已經(jīng)失敗了還是說考慮放棄這個(gè)任務(wù)(設(shè)置它的狀態(tài)為FAILED或ABANDONED)-這是業(yè)務(wù)邏輯層的事情,無法做到自動(dòng)決策。
只有在不可重啟的任務(wù)中才需要設(shè)置為FAILED狀態(tài),或者你知道重啟后數(shù)據(jù)還是有效的。Spring Batch Admin中有一系列工具JobService,用以取消正在進(jìn)行執(zhí)行的任務(wù)。
4、失敗一個(gè)job
失敗的job是可以重新啟動(dòng)的,因?yàn)樗臓顟B(tài)是FAILED,如果step2失敗,則返回一個(gè)EARLY TERMINATION的返回碼,step3也就不會(huì)執(zhí)行。否則繼續(xù)執(zhí)行step3
<step id="step1" parent="s1" next="step2"> <step id="step2" parent="s2"> <fail on="FAILED" exit-code="EARLY TERMINATION"/> <next on="*" to="step3"/> </step> <step id="step3" parent="s3">
5、結(jié)束一個(gè)job
已經(jīng)結(jié)束的job是不能重新啟動(dòng)的,因?yàn)樗臓顟B(tài)是COMPLETED。如果step2失敗了,則step3就不執(zhí)行了,該job也就COMPLETED,結(jié)束了。如果step2成功了,則繼續(xù)往下執(zhí)行step3。
<step id="step1" parent="s1" next="step2"> <step id="step2" parent="s2"> <end on="FAILED"/> <next on="*" to="step3"/> </step> <step id="step3" parent="s3">
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot高級(jí)配置之臨時(shí)屬性、配置文件、日志、多環(huán)境配置詳解
這篇文章主要介紹了SpringBoot高級(jí)配置之臨時(shí)屬性、配置文件、日志、多環(huán)境配置,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-02-02JAVA 實(shí)現(xiàn)二叉樹(鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu))
本篇文章主要介紹用JAVA 實(shí)現(xiàn)二叉樹,并提供實(shí)例.對(duì)二叉樹數(shù)據(jù)結(jié)構(gòu)很好的學(xué)習(xí)實(shí)踐,有需要的朋友可以參考下2016-07-07在Mac下IDEA安裝并使用protobuf方式(Java)
這篇文章主要介紹了在Mac下IDEA安裝并使用protobuf方式(Java),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11Java計(jì)算一個(gè)數(shù)加上100是完全平方數(shù),加上168還是完全平方數(shù)
這篇文章主要介紹了Java計(jì)算一個(gè)數(shù)加上100是完全平方數(shù),加上168還是完全平方數(shù),需要的朋友可以參考下2017-02-02使用mybatis-plus分頁出現(xiàn)兩個(gè)Limit的問題解決
在使用MyBatis-Plus進(jìn)行分頁查詢時(shí),可能會(huì)遇到查詢SQL中出現(xiàn)兩個(gè)limit語句的問題,這通常是由于在多個(gè)模塊中重復(fù)引入了MyBatis-Plus的分頁插件所導(dǎo)致的,下面就來介紹一下如何解決,感興趣的可以了解一下2024-10-10Springboot GET和POST請求參數(shù)獲取方式小結(jié)
Spring Boot GET和POST請求參數(shù)獲取是開發(fā)人員經(jīng)常需要解決的問題,本文主要介紹了Springboot GET和POST請求參數(shù)獲取方式小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09Java的GUI編程之列表和組合框的設(shè)計(jì)使用
這篇文章主要介紹了Java的GUI編程之列表和組合框的設(shè)計(jì)使用,是Java圖形界面編程中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-10-10