Java使用Redis来实现分布式锁

Java使用Redis来实现分布式锁

在单节点服务中,我们可以使用synchronized来保证同一时间内只允许一个线程执行限定的代码块。但是如果我们是多节点服务呢,因为synchronized是针对服务内部的,其他服务是无法受到他的干预的。那么如何保证多个节点在同一时间内只允许一个节点中的一个线程去访问这个代码块呢?

使用分布式锁!!!

本文使用Redisson来操作Redis并实现分布式锁

Redisson

官网:https://github.com/redisson/redisson/tree/redisson-3.16.8

  1. 引入依赖

    xml 复制代码
    <dependency>
       <groupId>org.redisson</groupId>
       <artifactId>redisson</artifactId>
       <version>3.16.8</version>
    </dependency> 
  2. 创建RedissonConfig配置类

    java 复制代码
    @ConfigurationProperties(prefix = "spring.redis")
    @Configuration
    @Data
    public class RedissonConfig {
        // 主机名
        private String host;
        // 端口
        private String port;
        // 使用那个数据库
        private Integer database;
    
        @Bean
        public RedissonClient redissonClient(){
            Config config = new Config();
            // 使用单机Redis服务
            config.useSingleServer()
                    // use "rediss://" for SSL connection
                    .setAddress(String.format("redis://%s:%s",host,port))
                    .setDatabase(database);
            return Redisson.create(config);
        }
    }
  3. 创建配置文件

    properties 复制代码
    # Redis 配置
    spring.redis.host=xxx
    spring.redis.port=xxx
    spring.redis.database=xxx
  4. 在业务代码中添加分布式锁

    java 复制代码
    // 注入我们的RedissonClient
    @Autowired
    private RedissonClient redissonClient;
    
    {
        // 指定一个key来获取锁,如果是需要按不同的情况加锁的话,这样可以使用变量
        RLock lock = redissonClient.getLock("LOCK_NAME");
        try{
            lock.tryLock(long waitTime, long leaseTime, TimeUnit unit);
            //... 业务逻辑
        }catch (InterruptedException ex) {
            ex.printStackTrace();
        }finally{
            // 判断是否由当前线程持有锁 
            if (lock.isHeldByCurrentThread()) {
                // 释放锁
                lock.unlock();
            }
        }
    }

    tryLock 尝试获取锁 获取成功返回true 获取失败返回false

    • waitTime 尝试获取锁的等待时间,超过不再继续获取
    • leaseTime 锁的持有时间,业务代码的执行时间如果超过该时间则抛出异常
      • 可以设置为null 或者 -1 表示业务执行多久占用多久,这里其实使用了看门狗的一个机制,默认的持有时间是30秒,如果超时未执行完成,每10看门狗会为我们自己续期一次。已保证业务代码执行完成。
    • unit 时间单位
相关推荐
伟大的大威1 天前
NVIDIA DGX Spark (Blackwell GB10) 双机 196B Step 3.5 Flash 大模型部署完整实录
分布式·spark·nvidia
HalvmånEver1 天前
7.高并发内存池大页内存申请释放以及使用定长内存池脱离new
java·spring boot·spring
凤山老林1 天前
SpringBoot 使用 H2 文本数据库构建轻量级应用
java·数据库·spring boot·后端
咖啡の猫1 天前
Redis桌面客户端
数据库·redis·缓存
赶路人儿1 天前
UTC时间和时间戳介绍
java·开发语言
dreamread1 天前
【SpringBoot整合系列】SpringBoot3.x整合Swagger
java·spring boot·后端
6+h1 天前
【java】基本数据类型与包装类:拆箱装箱机制
java·开发语言·python
what丶k1 天前
如何保证 Redis 与 MySQL 数据一致性?后端必备实践指南
数据库·redis·mysql
一直都在5721 天前
Spring面经
java·后端·spring
xiaoye37081 天前
如何在Spring中使用注解配置Bean的生命周期回调方法?
java·spring