Springboot整合Redission分布式锁使用实例

Springboot整合Redission分布式锁

引言:实际项目中,我们经常会遇到一些需要考虑使用分布式锁的场景,以防止页面重复请求或者多系统之间相互重复调用的产生业务偏差的问题;

例如:

1.并发的场景下,生成订单需要进行使用分布式锁来锁定商品库存,避免出现超卖情况。

2.定时任务,部署多个服务,存在某些任务被同时执行的情况;

一、springboot引入redission依赖;

java 复制代码
        <!-- redis 分布式锁 -->
        <dependency>
            <groupId>org.redisson</groupId>
            <artifactId>redisson</artifactId>
            <version>3.5.0</version>
        </dependency>

二、配置redis链接信息:实际各配置信息根据项目具体情况而定

java 复制代码
##默认密码为空
redis:
  host: 127.0.0.1
  port: 6379
  jedis:
    pool:
      #连接池最大连接数(使用负值表示没有限制)
      max-active: 100
      # 连接池中的最小空闲连接
      max-idle: 10
      # 连接池最大阻塞等待时间(使用负值表示没有限制)
      max-wait: 100000
  timeout: 5000
  database: 1

三、RedissonClient客户端配置(单机模式)

java 复制代码
import org.redisson.Redisson;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * 单机模式分布式锁配置
 */

@Configuration
public class RedissionConfig {

    @Value("${redis.host}")
    private String redisHost;

    @Value("${redis.port}")
    private String redisPort;

    @Value("${redis.database}")
    private int database;


    @Bean(destroyMethod = "shutdown")
    public RedissonClient redissonClient() {
        Config config = new Config();
        config.useSingleServer()
                .setAddress("redis://" + redisHost + ":" + redisPort)
                .setDatabase(database);
        return Redisson.create(config);
    }
}

四、测试

4.1 lockTest1方法对锁进行锁定,让现场睡眠1分钟,模拟业务一直占用锁的场景;

4.2 lockTest2 在业务1执行时候尝试去获取锁,看是否能获取到锁,同时等30秒后锁过期再去获取锁,是否可以正常获取到锁;

java 复制代码
	@Resource
    private RedissonClient redissonClient;

    @ApiOperation(value = "分布式可重入锁测试1", notes = "分布式锁测试1")
    @PostMapping("/lockTest1")
    Result lockTest1(@RequestBody RedisRequestDto dto){
        RLock lock = redissonClient.getLock(dto.getKey());
        String message = "";
        try {
            // lock.lock(); 会有看门狗机制,默认30秒;
//            lock.lock();

            // 此处采用手动设置过期时间,不会触发看门狗机制
            boolean locked = lock.tryLock(30, 60, TimeUnit.SECONDS);
            if(locked){
                message = "lockTest1 == 锁成功了";
                log.info(message);
            }else {
                message = "lockTest1 == 1锁失败了";
                log.info(message);
            }
            Thread.sleep(60000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            if(Objects.nonNull(lock) && lock.isHeldByCurrentThread()){
                log.info("{}锁被释放",dto.getKey());
                lock.unlock();
            }
        }
        return Result.ok(message);
    }

    @ApiOperation(value = "分布式可重入锁测试2", notes = "分布式锁测试2")
    @PostMapping("/lockTest2")
    Result lockTest2(@RequestBody RedisRequestDto dto){
        RLock lock = redissonClient.getLock(dto.getKey());
        try {
            boolean isLocked = lock.isLocked();
            if(isLocked){
                log.info("{}锁被占用,请稍后",dto.getKey());
                return Result.ok("锁占用,请稍后");
            }else {
                log.info("{}锁未占用,继续执行",dto.getKey());
                return Result.ok("锁未占用,继续执行");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            if(Objects.nonNull(lock) && lock.isHeldByCurrentThread()){
                lock.unlock();
            }
        }
        return Result.ok("");
    }

测试结果:

请求参数如下:

1.lockTest1 执行以后,我们可以看到有在db1中有一条记录,同时返回锁成功;

2.lockTest2 执行时候就是锁占用,证明"test_lock"已经被占用,无法获取;

3.等待30秒以后,test_lock锁自动过期,再次执行lockTest2,可以正常获取到锁;

通过日志,我们也能看出整个锁的锁定和释放的情况;

java 复制代码
2024-01-15 15:15:13.560  INFO 191576 --- [nio-1012-exec-1] c.s.house.controller.RedisController     : lockTest1 == 锁成功了
2024-01-15 15:15:16.960  INFO 191576 --- [nio-1012-exec-3] c.s.house.controller.RedisController     : test_lock锁被占用,请稍后
2024-01-15 15:16:24.566  INFO 191576 --- [nio-1012-exec-5] c.s.house.controller.RedisController     : test_lock锁未占用,继续执行
相关推荐
Wang's Blog2 分钟前
Elastic Stack梳理:Logstash Input插件详解与Codec插件应用指南之文件监控、多行日志处理与Kafka集成
分布式·搜索引擎·kafka·elastic search
klzdwydz2 分钟前
注解与反射
java·开发语言
哈哈老师啊9 分钟前
Springboot学生接送服务平台8rzvo(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
数据库·spring boot·后端
VX:Fegn08959 分钟前
计算机毕业设计|基于springboot + vue图书商城系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
talenteddriver11 分钟前
java: 分页查询(自用笔记)
java·开发语言
enjoy编程12 分钟前
Spring-AI 利用KeywordMetadataEnricher & SummaryMetadataEnricher 构建文本智能元数据
java·人工智能·spring
繁华似锦respect12 分钟前
lambda表达式中的循环引用问题详解
java·开发语言·c++·单例模式·设计模式·哈希算法·散列表
int WINGsssss28 分钟前
【无标题】
pytorch·分布式·python
heartbeat..29 分钟前
介绍一下软件开发中常见的几种的架构模式
java·架构·开发