基于Mongodb分布式锁简单实现,解决定时任务并发执行问题

前言

我们日常开发过程,会有一些定时任务的代码来统计一些系统运行数据,但是我们应用有需要部署多个实例,传统的通过配置文件来控制定时任务是否启动又太过繁琐,而且还经常出错,导致一些异常数据的产生

网上有很多分布式锁的实现方案,基于redis、zk、等有很多,但是我的就是一个用了mysql和mongo的小应用,不准备引入其他三方中间件来解决这个问题,撸一个简单的分布式锁来解决定时任务并发执行的问题,加锁操作的原子性和防死锁也都要支持,这里我使用mongodb写了AllInOne的工具类

All in one Code

先上代码

@Component
@Slf4j
public class MongoDBLock {

    private static final int DEFAULT_LOCK_TIMEOUT = 30;//锁的默认超时时间,单位秒

    private MongoTemplate mongoTemplate;
    private int lockTimeout;

    public MongoDBLock(MongoTemplate mongoTemplate) {
        this.mongoTemplate = mongoTemplate;
        this.lockTimeout = DEFAULT_LOCK_TIMEOUT;
    }

    /**
     * 尝试获取分布式锁
     *
     * @param lockKey 锁的key
     * @return true:获取锁成功,false:获取锁失败
     */
    private boolean acquireLock(String lockKey) {
        LockDocument document = new LockDocument();
        document.setId(lockKey);
        document.setExpireAt(Instant.ofEpochMilli(Instant.now().toEpochMilli() + lockTimeout * 1000));
        try {
            mongoTemplate.insert(document);
            return true;
        } catch (Exception e) {

        }
        return false;
    }

    /**
     * 释放分布式锁
     *
     * @param lockKey 锁的key
     */
    private void releaseLock(String lockKey) {
        Query query = new Query(Criteria.where("key").is(lockKey));
        mongoTemplate.remove(query, LockDocument.class);
        log.info("程序执行成功,释放分布式锁,lockKey:{}",lockKey);
    }

    /**
     * 分布式锁入口方法,参数lockName为锁的名称,lockKey为需要加锁的key,执行完成后自动释放锁
     *
     * @param lockKey
     * @param task
     * @param <T>
     * @throws Exception
     */
    public <T> void executeWithLock(String lockKey, ITask<T> task) throws Exception {
        boolean locked = acquireLock(lockKey);
        if (locked) {
            log.info("获取分布式锁成功,lockKey:{}",lockKey);
            try {
                task.execute();
            } finally {
                releaseLock(lockKey);
            }
        } else {
            log.warn("获取分布式锁失败,lockKey:{}", lockKey);
            throw new AppException("获取分布式锁失败!");
        }
    }

    @Data
    @Document(collection = "lock_collection")
    static class LockDocument {
        @Id
        private String id;
        @Indexed(expireAfterSeconds = DEFAULT_LOCK_TIMEOUT)
        private Instant expireAt;
    }

    @FunctionalInterface
    public interface ITask<T> {
        T execute() throws Exception;
    }
}

调用示例

    @Resource
    MongoDBLock mongoDBLock;

    mongoDBLock.executeWithLock("key", () -> {
        // do some thing
        return null;
    });

原理

  • 使用key作为主键,利用mongodb的insert原子性保障LockDocument不会重复插入
  • LockDocument中expireAt字段利用的mongodb索引过期机制,解决死锁问题,这里设置超时时间是30秒,并在执行完成之后会主动释放锁

原文链接:https://www.cnblogs.com/surging-dandelion/p/17330389.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:基于Mongodb分布式锁简单实现,解决定时任务并发执行问题 - Python技术站

(0)
上一篇 2023年4月19日
下一篇 2023年4月19日

相关文章

  • Java中Arrays类与Math类详解

    Java中Arrays类与Math类详解 在Java中,Arrays类和Math类是常用的工具类,主要提供了一些静态方法来方便我们进行数组、数值计算等操作。 Arrays类 Arrays类提供了很多有用的方法来进行数组的操作,包括数组的排序、查找、复制等。 数组排序 排序算法 Arrays类中提供了sort()方法来对数组进行排序,在方法中我们可以通过传入C…

    Java 2023年5月26日
    00
  • Java基础之FastJson详解

    Java基础之FastJson详解 FastJson是一个Java语言编写的轻量级JSON解析工具,具有解析速度快、易用性好等优点。本文将从以下几个方面详细讲解FastJson的使用: 导入FastJson依赖 基本用法 使用注解进行自定义序列化与反序列化 高级特性 导入FastJson依赖 在使用FastJson之前,我们需要在项目中导入FastJson依…

    Java 2023年5月26日
    00
  • java暴力匹配及KMP算法解决字符串匹配问题示例详解

    Java暴力匹配及KMP算法解决字符串匹配问题 1. 概述 在字符串匹配问题中,有两种经典的算法:暴力匹配和KMP算法。暴力匹配是最简单的字符串匹配算法,其思路是将字符串的每个子串与目标字符串进行匹配。KMP算法是一种更高效的字符串匹配算法,它通过预处理字符串的next数组来避免不必要的字符比较,从而在匹配过程中提高效率。 2. Java暴力匹配 暴力匹配算…

    Java 2023年5月19日
    00
  • Java SpringBoot自动配置原理详情

    Java SpringBoot是目前非常流行的开源框架之一,它通过自动配置的方式简化了开发者的工作量,并提高了开发效率。本文将详细讲解Java SpringBoot自动配置的原理以及实际应用,让读者对SpringBoot有更深刻的理解。 SpringBoot自动配置原理 SpringBoot自动配置的核心在于@EnableAutoConfiguration注…

    Java 2023年5月19日
    00
  • Java实现的串口通信功能示例

    为了实现串口通信功能,Java提供了一个称为Java Comm API的标准扩展。下面是实现Java串口通信的步骤: 下载并安装Java Comm API。Java Comm API不是JDK的一部分,需要单独下载、安装和配置。它提供了一个称为javax.comm的包,它包含用于访问串口的类和方法。 确定要使用的串口。您需要查看串口通信设备管理器,以查找可用…

    Java 2023年5月19日
    00
  • javascript+html5+css3自定义弹出窗口效果

    “javascript+html5+css3自定义弹出窗口效果”主要可以通过以下步骤实现: 第一步:HTML部分 在HTML代码中,我们需要首先定义一个触发弹出窗口的按钮,用于触发弹出窗口的开启。同时我们需要在代码中定义一个 <div> 标签作为弹出窗口的容器。 示例代码如下: <!DOCTYPE html> <html>…

    Java 2023年6月15日
    00
  • jsp编程获取当前目录下的文件和目录及windows盘符的方法

    获取当前目录下的文件和目录方法一般分为两步: 获取当前目录的绝对路径。 遍历当前目录,获取其中所有的文件和目录。 具体的步骤如下: 1. 获取当前目录的绝对路径 在 JSP 中,可以通过 pageContext 对象的 getServletContext() 方法获取 ServletContext 对象,然后通过 getRealPath() 方法获取当前应用…

    Java 2023年6月15日
    00
  • java实现纸牌游戏之小猫钓鱼算法

    Java实现纸牌游戏之小猫钓鱼算法攻略 简介 小猫钓鱼是一种流行的纸牌游戏,玩家需要通过牌面大小的推理和计算,最终获得胜利。本文将详细讲解Java实现小猫钓鱼游戏的算法步骤和代码实现。 算法步骤 初始化一副扑克牌,包括54张牌(52张普通牌和2张王牌); 洗牌,打乱牌的顺序; 将牌平均分成四份,分别给四个玩家,并根据牌面大小进行排序,从大到小排列; 小猫玩家…

    Java 2023年5月19日
    00
合作推广
合作推广
分享本页
返回顶部