spring定时任务scheduler集群环境下防止多次执行

方式一:
使用spring的@Scheduler注解可以非常方便的启动一个定时任务,但是当服务部署在多台服务器上做负载均衡的时候,可能会出现重复执行的情况。现在我们通过代码指定job只在某一台机器执行。首先在系统中配置一个执行job的机器的ip地址

在这里插入图片描述

然后,当job开始的时候就将本地ip和配置的ip对比,如果不是的话就结束任务。

spring定时任务scheduler集群环境下防止多次执行_第1张图片

方式二:
在项目中使用@Scheduled注解实现定时调度十分方便,但是存在一个情况:当项目多节点部署时,调度会在每个节点上都执行。这就是个大问题了。网上解决方案也比较多,我用过两种方案:第一种是:指定一台主机,让调度只在一台主机上执行。第二种是:使用redis的分布式锁实现。

第一种,失去了分布式部署的宗旨。下面我简单的介绍下第二种

1、创建Redis工具类中,增加setNX方法,即"SET if Not Exists",成功返回true,失败返回flase:

   public Boolean setNX(String key, String value,long timeout, TimeUnit unit) {
        Boolean isExit = this.redisTemplate.getConnectionFactory().getConnection().setNX(key.getBytes(), value.getBytes());
        //如果设置成功,要设置其过期时间
        if (isExit) {
            redisTemplate.expire(key, timeout, unit);
        }
        return isExit;
    }

2、在业务代码中引用方法:

public void testtest() throws InterruptedException {
    try{
       if (!redisUtil.setNX("quartzFlag","1",30,TimeUnit.SECONDS)){//设置超时时间
            LOGGER.info("任务已执行");
            return;
        }
        
        业务代码....
        ...........
     } catch (Exception e) {
            e.printStackTrace();
        } finally {
            //任务完成后,释放redis锁
            RedisUtil.del("activityScheduledFlag");
      }

为防止程序意外终止,导致死锁,文章中用到了,任务完成释放锁和设置过期时间双重保障。

链接:https://www.jianshu.com/p/30c1fb5f7f04
https://blog.csdn.net/No_overtime_apes/article/details/90230307

你可能感兴趣的:(spring定时任务scheduler集群环境下防止多次执行)