集群模式,如何保证@Scheduled定时任务只执行一次

问题

假如我们有一个定时任务,每天凌晨0点,给当天过生日的人,积分加1

@Scheduled(cron = "0 0 0 */1 * * *")
public void updateScore() {
    //doSomething...
}

单主机部署下不会有任何问题。

但是如果有多台主机,假设我们有4台,那么这个定时任务就会执行4次,造成积分+4

解决方法

解决方法有很多种:redis + token,redis + 分布式锁,这里我们采用乐观锁

数据库新建一张表:

CREATE TABLE `t_lock` (
  `lock_key` varchar(15) NOT NULL COMMENT '定时任务Key',
  `utime` bigint(20) NOT NULL COMMENT '用于乐观锁的时间',
  UNIQUE KEY `lockUniq` (`lock_key`,`utime`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

一个项目里可能有多个定时任务,lock_key用来区分定时任务,utime用于更新操作。

每天凌晨0点,utime传入当天时间(初始值设一个很小的值,保证第一次可以执行),然后执行sql语句:

update
      `t_lock`
set
       `utime` = #{utime}
where
       `lock_key`=#{lock_key} and `utime`<#{utime}

此时4台主机的这个定时任务,只有1台主机执行update成功,并且返回1,其他都返回0,在返回0的时候,直接return结束

@Scheduled(cron = "0 0 0 */1 * * *")
public void updateScore() {
	//tLock对象传入当前时间和key
	...
	//获取update影响行数
    int i = tLockMapper.update(tLock);
    if(i == 0) {
        return;
    }
    //doSomething...
}
  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
@Scheduled定时任务执行两次的原因可能有多种,根据引用内容,其中可能的原因之一是服务器配置的问题。以Tomcat服务器为例,如果在配置文件server.xml中同时配置了appBase="webapps"和docBase="YourApp",相当于加载了两个应用,导致定时任务执行两次。解决方案之一是修改配置文件,确保只加载一个应用。另外,还可能涉及其他因素,比如代码中的依赖关系、任务管理器的配置等。需要进一步检查和排查相关配置和代码以确定具体原因。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [生产问题:@Scheduled Spring定时任务每次执行两次原因分析以及解决方案](https://blog.csdn.net/u010886217/article/details/107429853)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [简洁的定时任务实例](https://download.csdn.net/download/qq_37173966/9996155)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [SpringBoot @Scheduled定时任务执行两次问题](https://blog.csdn.net/ddzyx/article/details/113240322)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值