spring-integration-redis中分布式锁RedisLockRegistry的使用

5 篇文章 0 订阅
4 篇文章 0 订阅

pom依赖:

<!-- redis -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.integration</groupId>
    <artifactId>spring-integration-redis</artifactId>
</dependency>

配置:

@Configuration
public class RedisLockConfig {
 
    @Bean
    public RedisLockRegistry redisLockRegistry(RedisConnectionFactory redisConnectionFactory) {
         //第一个参数redisConnectionFactory
         //第二个参数registryKey,分布式锁前缀,设置为项目名称会好些
         //该构造方法对应的分布式锁,默认有效期是60秒.可以自定义
         return new RedisLockRegistry(redisConnectionFactory, "boot-launch");
         //return new RedisLockRegistry(redisConnectionFactory, "boot-launch",60);
     }
}

方式1:


    @Autowired
    private RedisLockRegistry redisLockRegistry;
Lock lock = redisLockRegistry.obtain("lockkey-" + 1);
//tryLock尝试获取锁(成功返回true,失败返回false);
    if (lock.tryLock()) {
        try {
            // 业务代码块
 
        } finally {
        //走到这里说明获取锁一定是成功的
        try{
        lock.unlock();
        }catch (IllegalStateException e){
        //业务操作时间过长,会导致锁超时内系统释放,这里就会产生超时异常
        log.error("释放时发现锁过期");
        }
            
        }
    } else {
    //走到这里说明获取锁一定是失败的的
        log.info("并发抢锁失败");
    }

方式2:


    @Autowired
    private RedisLockRegistry redisLockRegistry;
Lock lock = redisLockRegistry.obtain("lockkey-" + 1);
//lock()一直循环获取锁,循环周期为100毫秒(成功了则无返回,失败了则抛异常),理论上一定能获取到锁,因为会一直循环取锁
    try {
    lock.lock();
        // 业务代码块(这里要判断并发情况下的业务幂等性)
    }catch (CannotAcquireLockException e){
    //当redis挂掉时执行redis操作失败会产生异常CannotAcquireLockException;说明获取锁失败了
    log.error("get redis lock fail");
    } finally {
    //走到这里 获取锁不一定是成功的,异常时也走,因此需要判断lock状态
    if(Objects.nonNull(lock)){
    //锁存在才需要释放
        try{
        lock.unlock();
        }catch (IllegalStateException e){
        //业务操作时间过长,会导致锁超时内系统释放,这里就会产生超时异常
        log.error("释放时发现锁过期");
        }
    }
    }

方式3:


    @Autowired
    private RedisLockRegistry redisLockRegistry;
Lock lock = redisLockRegistry.obtain("lockkey-" + 1);
try {
//tryLock(3, TimeUnit.SECONDS):在3秒钟内循环获取锁,超过3秒还没有获取到就认为获取失败了,这个方法会抛出异常InterruptedException,
        boolean isLocked = lock.tryLock(3, TimeUnit.SECONDS);
        if (isLocked) {
        // 业务代码块
    }
        } catch (InterruptedException e) {
            log.error("get redis lock fail");
        }finally {
        //获取锁失败的时候也会走,需要判断锁失败时不去释放
            if(Objects.nonNull(lock)){
    //锁存在才需要释放
        try{
        lock.unlock();
        }catch (IllegalStateException e){
        //业务操作时间过长,会导致锁超时内系统释放,这里就会产生超时异常
        log.error("释放时发现锁过期");
        }
    }
        }

总结:

解决同一个用户并发操作同一个资源时,如提交时按钮连击、多端调用同一个业务接口:使用方式1;方式2(使用方式2时需要做幂等判断,如连击操作多次调用时,对第二次调用不做处理)

解决并发限流时,如抽奖系统中对商品sku库存做锁操作,防止多个用户同时扣减库存导致库存数量少扣:使用方式2、或者方式3(锁等待时间=预计业务代码执行时间)

三方式都要注意释放锁时做锁过期处理

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
RedisLockRegistrySpring Integration 提供的一种基于 Redis分布式锁实现。它使用 Redis 实现了锁的管理和协调,提供了一种简单而有效的方式来确保在分布式环境的资源互斥访问。 使用 RedisLockRegistry 首先需要配置 Redis 连接信息,然后通过创建 RedisLockRegistry 对象来获取分布式锁实例。下面是一个简单的示例: ```java @Configuration @EnableIntegration public class RedisLockConfig { @Bean public RedisConnectionFactory redisConnectionFactory() { // 配置 Redis 连接工厂 RedisStandaloneConfiguration config = new RedisStandaloneConfiguration("localhost",6379); return new LettuceConnectionFactory(config); } @Bean public RedisLockRegistry redisLockRegistry(RedisConnectionFactory redisConnectionFactory) { // 创建 RedisLockRegistry 实例,并指定 Redis 连接工厂 return new RedisLockRegistry(redisConnectionFactory, "my-locks"); } } ``` 在需要使用分布式锁的地方,可以通过 RedisLockRegistry 来获取锁并执行相应的操作。下面是一个简单的示例: ```java @Autowired private RedisLockRegistry redisLockRegistry; public void performTaskWithLock() { Lock lock = redisLockRegistry.obtain("my-lock"); // 获取名为 "my-lock" 的分布式锁 try { if (lock.tryLock()) { // 尝试获取锁 // 执行需要互斥访问的操作 } else { // 锁被其他线程占用,执行相应的逻辑 } } finally { lock.unlock(); // 释放锁 } } ``` RedisLockRegistry 会在 Redis 创建相应的键来表示锁的状态,并使用 Redis 的原子操作来确保锁的正确获取和释放。这样可以保证在分布式场景下的资源互斥访问。 需要注意的是,在使用 RedisLockRegistry 进行分布式锁管理时,需要确保 Redis 的可用性和性能,以及合理设置锁的超时时间和重试机制,以防止死锁和长时间的资源占用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值