1.场景描述

老项目需要多机部署,项目中有几十个定时任务,一旦多机部署,定时任务就会重复执行,固定ip与错开时间方案都存在较大弊端,最终采用的方案是:AOP+排他锁的方式,软件老王已验证通过,介绍下,有需要的朋友可以参考下。

2.解决方案

软件老王基本方案是采用:AOP+排他锁的方式。

(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;

}

}

说明:

(1)面向标签编程

@Pointcut( "@annotation(org.springframework.scheduling.annotation.Scheduled)")

这样注解会拦截标签@Scheduled。

(2)使用aop的环绕标签 @Around("scheduled()")

@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 t_schedule_cluster

execute_ip = #{executeIp,jdbcType=VARCHAR},

version = #{version,jdbcType=INTEGER} + 1,

execute = #{execute,jdbcType=INTEGER},

status = #{status,jdbcType=VARCHAR},

createby = #{createby,jdbcType=VARCHAR},

createdate = #{createdate,jdbcType=TIMESTAMP},

updateby = #{updateby,jdbcType=VARCHAR},

updatedate = #{updatedate,jdbcType=TIMESTAMP},

where task_name = #{taskName,jdbcType=VARCHAR}

and version = #{version,jdbcType=INTEGER}

and status ='1'

(2)第二个是释放锁,更改excute为0。

update t_schedule_cluster

execute_ip = #{executeIp,jdbcType=VARCHAR},

version = #{version,jdbcType=INTEGER},

execute = #{execute,jdbcType=INTEGER},

status = #{status,jdbcType=VARCHAR},

createby = #{createby,jdbcType=VARCHAR},

createdate = #{createdate,jdbcType=TIMESTAMP},

updateby = #{updateby,jdbcType=VARCHAR},

updatedate = #{updatedate,jdbcType=TIMESTAMP},

where task_name = #{taskName,jdbcType=VARCHAR}

and status ='1'

I’m 「软件老王」,如果觉得还可以的话,关注下呗,后续更新秒知!欢迎讨论区、同名公众号留言交流!

关于找一找教程网

本站文章仅代表作者观点,不代表本站立场,所有文章非营利性免费分享。

本站提供了软件编程、网站开发技术、服务器运维、人工智能等等IT技术文章,希望广大程序员努力学习,让我们用科技改变世界。

[多机部署之定时任务完整方案]http://www.zyiz.net/tech/detail-97083.html

java 多定时任务_多机部署之定时任务完整方案相关推荐

  1. java 时间轮_基于时间轮的定时任务

    JobScheduleHelper是一个单列. 1-来看看类中的一些属性: public static final long PRE_READ_MS = 5000; // pre read priva ...

  2. android原生定时任务_一个简单的Android定时任务

    Android中(Service )服务的最佳实践--后台执行的定时任务 版权声明:本文为博主原创文章,未经博主允许不得转载.https://blog.csdn.net/u010046908/arti ...

  3. xxljob 配置具体定时任务_记一次xxl-job定时任务没有触发的问题

    当初选了xxl-job就是因为它的触发机制比较靠谱,到点准时发,而且有日志可以看. 昨天突然发现部署在一台本地机器上的xxl-job到点并没有触发,且没有任何日志.通过管理页面查询触发日志,发现日志还 ...

  4. java定时任务_定时任务最简单的3种实现方法(超好用)

    定时任务在实际的开发中特别常见,比如电商平台 30 分钟后自动取消未支付的订单,以及凌晨的数据汇总和备份等,都需要借助定时任务来实现,那么我们本文就来看一下定时任务最简单的几种实现方式. TOP 1: ...

  5. java 客户端定时任务_定时任务最简单的3种实现方法(超实用)

    定时任务在实际的开发中特别常见,比如电商平台 30 分钟后自动取消未支付的订单,以及凌晨的数据汇总和备份等,都需要借助定时任务来实现,那么我们本文就来看一下定时任务最简单的几种实现方式. TOP 1: ...

  6. java获取mysql数据定时执行_详解SpringBoot 创建定时任务(配合数据库动态执行)...

    序言:创建定时任务非常简单,主要有两种创建方式:一.基于注解(@Scheduled) 二.基于接口(SchedulingConfigurer). 前者相信大家都很熟悉,但是实际使用中我们往往想从数据库 ...

  7. xxljob 配置具体定时任务_快速使用分布式定时任务 xxl-job

    快速使用分布式定时任务 xxl-job 需要linux服务器环境安装: jdk1.8 ,docker 安装步骤地址 : linux快速安装jdk   . 在linux里安装docker 1.docke ...

  8. Java堆空间,本机堆和内存问题

    最近,我在和一个朋友讨论为什么Java进程使用的内存比启动Java进程时设置的最大堆多. 代码创建的所有Java对象都是在Java堆空间内创建的,其大小由-Xmx选项定义. 但是一个Java进程由很多 ...

  9. Fabric实战(四)-多机部署fabric网络-solo

    ####多机部署fabric(solo模式) 本案例为: 一个orderer节点 两个peer节点 三台物理机: orderer:192.168.2.148peer0org1:192.168.2.10 ...

  10. Java微服务_医疗管理项目_基于若依快速开发框架

    一.项目简介 项目简介:尚医疗是专门为各大医院.门]诊提供的一款医疗管理平台.系统包含:系统 管理.药品进销存管理.看病就诊.收费管理.检查管理.数据统计等核心模块.通过尚医 疗系统可以快速方便的管理 ...

最新文章

  1. 智源学者朱军获2020年“科学探索奖”
  2. Redis系列四:redis支持的数据类型
  3. 假设以带头结点的循环链表表示队列_[leetcode链表系列]2 删除链表中的节点
  4. 解决执行go get时报错的问题:dial tcp: lookup xxx.com on 8.8.8.8:53: no such host
  5. HDU1520 Anniversary party 树形动态规划
  6. sudo: apt-get:找不到命令_Linux重复执行历史命令方法详解
  7. 安装Pywin32后无法正常引用pyd文件
  8. python 相对导入_python 相对导入与绝对导入
  9. c++ 副本构造器
  10. Linux 配置网络
  11. Centos8 加密 GRUB 防破解root密码
  12. 【转】TIOBE 12 月排行榜:古老的 C 和后起之秀 Kotlin,谁是年度编程语言之王?...
  13. 实验二(第一部分,第二部分)
  14. 读书笔记《松本行弘的程序世界》
  15. 全家桶靠边站 用Windows自带功能解决3大难题
  16. leetcode75 颜色分类
  17. 一路向西17天之 乌市-布尔津-空中花园-禾木
  18. Oracle 11g 学习笔记-14(数据库性能优化)
  19. 浙江科技大学计算机调剂,浙江科技大学计算机应用技术专业2019年考研调剂信息...
  20. 程序设计与c语言笔记(一)

热门文章

  1. python 输出 百分比
  2. 网站死链查询检测方法(seo的优化工作全攻略)
  3. 笔记︱盘点实验科学的三种实验模型(A/B实验、因果推断、强化学习)
  4. Influx Sql系列教程三:measurement 表
  5. Android P cdd(Android 兼容性定义文档)
  6. 达叔走了,别只发声感叹就完了
  7. 《高等数学B(一)》笔记
  8. WEEK 7作业 A-TT的魔法猫 B-TT的旅行日记 C-TT的美梦
  9. Protocol buffer配置-生成jar包和java文件
  10. SqlServer执行计划