1.場景描述
老項目須要多機部署,項目中有幾十個定時任務,一旦多機部署,定時任務就會重複執行,固定ip與錯開時間方案都存在較大弊端,最終採用的方案是:AOP+排他鎖的方式,軟件老王已驗證經過,介紹下,有須要的朋友能夠參考下。html
2.解決方案
軟件老王基本方案是採用:AOP+排他鎖的方式。spring
(1)目前老項目有幾十個定時任務,採用AOP的方式,能夠保證代碼的無侵入(即便簡單的微侵入,例如增長几行代碼,測試驗證的工做量也會比較大的)。 (2)採用排他鎖的方式,保證批處理的高可用,不重複執行。數據庫
2.1 AOP編程
Aop的概念就不說了,就是面向切面編程,通俗點就是統一處理一類問題,好比日誌、請求鑑權等,剛開始不肯定是否可行,系統中的批處理是使用spring註解的方式@Scheduled進行批處理,採用aop對註解@Scheduled進行編程,統一攔截批處理,代碼以下:編程
/** * 軟件老王-AOP處理類 */ @Aspect @Component public class ScheduledAspect { @Autowired ScheduleService scheduleService ; @Pointcut( "@annotation(org.springframework.scheduling.annotation.Scheduled)") public void scheduled() { } @Around("scheduled()") public Object scheduled(ProceedingJoinPoint pjd) { Object result = null; String taskName = pjd.getSignature().getName(); try { if (scheduleService.isInvoke(taskName)){ return result; } result = pjd.proceed(); scheduleService.end(taskName); } catch (Throwable e) { throw new RuntimeException(e); } return result; } }
說明:mybatis
(1)面向標籤編程app
@Pointcut( "@annotation(org.springframework.scheduling.annotation.Scheduled)")
這樣註解會攔截標籤@Scheduled。測試
(2)使用aop的環繞標籤 @Around("scheduled()")spa
@before標籤拿不到執行完成狀態,須要使用環繞標籤@@Around,在標籤中能夠拿到執行完成後狀態,以便放開鎖。設計
result = pjd.proceed();
(3)結合排他鎖使用日誌
@Autowired ScheduleService scheduleService ;
2.2 排他鎖
排他鎖,簡單來講就是經過數據庫總的標誌位+版版號進行的控制.
軟件老王的代碼以下,:
/** * 軟件老王-排他鎖服務類 */ @Service public class ScheduleService { @Autowired ScheduleClusterMapper scheduleClusterMapper; public boolean isInvoke(String taskName) { boolean isValid = false; try { ScheduleCluster carIndexEntity = scheduleClusterMapper.selectByTaskName(taskName); int execute = carIndexEntity.getExecute(); String ip = InetAddress.getLocalHost().getHostAddress(); long currentTimeMillis = System.currentTimeMillis(); long time = carIndexEntity.getUpdatedate().getTime(); if (execute == 0) { isValid = start(taskName, carIndexEntity.getVersion(), ip); } } catch (UnknownHostException e) { e.printStackTrace(); } return isValid; } //執行鎖機制,軟件老王 public boolean start(String taskName, int version, String ip) { ScheduleCluster scheduleCluster = new ScheduleCluster(); scheduleCluster.setVersion(version); scheduleCluster.setExecuteIp(ip); scheduleCluster.setUpdatedate(DateUtil.getCurrentTime()); scheduleCluster.setTaskName(taskName); scheduleCluster.setExecute(1); int count = scheduleClusterMapper.updateByTaskName(scheduleCluster); if (count > 0) { return true; } return false; } //執行解鎖機制,軟件老王 public void end(String taskName) { ScheduleCluster scheduleCluster = new ScheduleCluster(); scheduleCluster.setUpdatedate(DateUtil.getCurrentTime()); scheduleCluster.setTaskName(taskName); scheduleCluster.setExecute(0); scheduleClusterMapper.updateNormalByTaskName(scheduleCluster); } }
說明:
大的原理是在where條件後帶上版本號,在update中更新version+1,這樣經過影響數據庫的影響條數,來判斷是否拿到鎖。
(1)主類中調用start方法,該方法是更新批處理狀態,軟件老王這裏設置了一個小點,在updateByTaskName的mybatis方法中,有個version+1的更新;
(2)end方法放在更新完成後,釋放鎖。
(3)其實還有一個點,能夠考慮下,須要有個機制,好比出現異常狀況,恰好批處理執行中,重啓服務了等,下次批處理執行前,假如鎖還未釋放,代碼中增長釋放鎖的機制。
2.3 數據庫相關
(1)數據庫表設計
(2)mybatis相關方法
(1)第一個是start對應方法,執行鎖和version增長。
<update id="updateByTaskName" parameterType="com.yutong.dmp.entity.ScheduleCluster"> update t_schedule_cluster <set> <if test="executeIp != null"> execute_ip = #{executeIp,jdbcType=VARCHAR}, </if> <if test="version != null"> version = #{version,jdbcType=INTEGER} + 1, </if> <if test="execute != null"> execute = #{execute,jdbcType=INTEGER}, </if> <if test="status != null"> status = #{status,jdbcType=VARCHAR}, </if> <if test="createby != null"> createby = #{createby,jdbcType=VARCHAR}, </if> <if test="createdate != null"> createdate = #{createdate,jdbcType=TIMESTAMP}, </if> <if test="updateby != null"> updateby = #{updateby,jdbcType=VARCHAR}, </if> <if test="updatedate != null"> updatedate = #{updatedate,jdbcType=TIMESTAMP}, </if> </set> where task_name = #{taskName,jdbcType=VARCHAR} and version = #{version,jdbcType=INTEGER} and status ='1' </update>
(2)第二個是釋放鎖,更改excute爲0。
<update id="updateNormalByTaskName" parameterType="com.yutong.dmp.entity.ScheduleCluster"> update t_schedule_cluster <set> <if test="executeIp != null"> execute_ip = #{executeIp,jdbcType=VARCHAR}, </if> <if test="version != null"> version = #{version,jdbcType=INTEGER}, </if> <if test="execute != null"> execute = #{execute,jdbcType=INTEGER}, </if> <if test="status != null"> status = #{status,jdbcType=VARCHAR}, </if> <if test="createby != null"> createby = #{createby,jdbcType=VARCHAR}, </if> <if test="createdate != null"> createdate = #{createdate,jdbcType=TIMESTAMP}, </if> <if test="updateby != null"> updateby = #{updateby,jdbcType=VARCHAR}, </if> <if test="updatedate != null"> updatedate = #{updatedate,jdbcType=TIMESTAMP}, </if> </set> where task_name = #{taskName,jdbcType=VARCHAR} and status ='1' </update>
I’m 「軟件老王」,若是以爲還能夠的話,關注下唄,後續更新秒知!歡迎討論區、同名公衆號留言交流!
原文出處:https://www.cnblogs.com/ruanjianlaowang/p/12053817.html