基于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 实战项目之疫情防控管理系统详解

    Java 实战项目之疫情防控管理系统详解 1. 项目介绍 该项目是一个基于Java的疫情防控管理系统。通过该系统,用户可以实现疫情病例的查询、疫情防控信息的发布和员工健康信息的管理等功能。 2. 技术栈 2.1 前端技术栈 HTML/CSS/JavaScript jQuery Bootstrap 2.2 后端技术栈 Java Spring/Spring MV…

    Java 2023年5月23日
    00
  • 剑指Offer之Java算法习题精讲数组与字符和等差数列

    剑指Offer之Java算法习题精讲数组与字符和等差数列 在剑指Offer面试题中,数组和等差数列相关的算法习题十分常见,该攻略将针对这些习题进行详细的讲解。 数组 在Java中,数组是一种非常基础的数据类型,它可以存储一组具有相同类型的数据。数组的下标从0开始,可以使用array[index]的方式获取数组中特定下标的元素。下面讲解两道涉及数组的算法题: …

    Java 2023年5月19日
    00
  • java对象拷贝详解及实例

    首先我们需要明确一下,Java中的对象拷贝指的是拷贝一个对象的副本,而不是两个对象共享同一块内存地址。在Java中,我们可以使用浅拷贝和深拷贝两种方式来实现对象的拷贝。 浅拷贝 浅拷贝最简单的方式就是使用Object类的clone()方法,该方法可以复制一个Java对象。但是,它并不是完全的复制。当我们使用clone()方法来复制一个Java对象时,它会返回…

    Java 2023年5月26日
    00
  • springboot 如何配置多个jndi数据源

    下面是详细讲解: 配置多个JNDI数据源 在Spring Boot中,可以使用application.properties或application.yml文件配置多个JNDI数据源。下面详细说明如何实现操作。 添加依赖 首先,需要在pom.xml文件中添加JNDI的支持依赖,例如: <dependency> <groupId>org.…

    Java 2023年5月20日
    00
  • 一小时迅速入门Mybatis之bind与多数据源支持 Java API

    下面是关于“一小时迅速入门 Mybatis 之 bind 与多数据源支持 Java API”的完整攻略: 什么是 Mybatis Mybatis 是一款优秀的 Java ORM 框架,它的主要作用是将数据库表映射成 Java 对象。在 Mybatis 中,可以通过 XML 或 Java 注解的方式进行数据映射,同时提供了丰富的 SQL 语句拼接、缓存、事务控…

    Java 2023年5月20日
    00
  • 什么是元空间?

    以下是关于 Java 元空间的详细讲解和使用攻略: 元空间的作用是什么? Java 元空间(Metaspace)是一种用于存储加载类信息、常量、静态变量、即时编译后的代码数据的内存区域。元空间是线程共享的,其大小可以通过 -XX:MaxMetaspaceSize 参数进行设置。 元空间的使用攻略 使用 Java 元空间,需要注意几点: 在程序开发中需要合理使…

    Java 2023年5月12日
    00
  • 深入理解Java8新特性之Lambda表达式的基本语法和自定义函数式接口

    深入理解Java8新特性之Lambda表达式的基本语法和自定义函数式接口 1. Lambda表达式的基本语法 Lambda表达式是一种匿名函数,可以在Java8及以后版本中使用。它可以将函数作为方法参数,或者将其作为数据来处理。 Lambda表达式的基本语法如下: (parameter1, parameter2, …, parameterN) ->…

    Java 2023年5月26日
    00
  • springboot项目启动慢的问题排查方式

    当Spring Boot项目启动慢或者无法正常启动时,可以采用以下排查方式: 一、查看启动日志输出,分析问题 启动Spring Boot项目时,可以查看控制台输出的启动日志,了解项目启动时的具体情况。如果日志中有异常或者错误信息,可以根据异常信息进行问题排查。 查看日志的方式有两种: 直接查看控制台输出的日志信息。 将日志输出到文件中,可以通过配置日志级别、…

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