Springboot使用Redis实现分布式锁

1、使用场景和实现方案:

使用场景:本地锁如Lock和Syncronized只能锁住本地进程,在分布式应用中,需要使用分布式锁来更好实现特定的业务。

实现方案:有多种,比如使用mysql、zookeeper、redis,各种方案有其优缺点,使用哪种方式需结合具体业务来选择。下面介绍一下使用redis实现分布式锁。

实现方式:如下图所示

具体实现代码:

2.1 使用redisTemplate:

java 复制代码
    public Map<String,  List<Catelog2Vo>> getCatelogJsonFromDbWithRedisLock()  {
        //使用redis分布式锁,防止缓存击穿
        String uuid = UUID.randomUUID().toString();
        Boolean lock = redisTemplate.opsForValue().setIfAbsent("lock", uuid,300,TimeUnit.SECONDS);
        Map<String, List<Catelog2Vo>>map;

        if(lock){
            //如果加锁成功,执行业务
            System.out.println("获取分布式锁成功...........");
            map = getDataFromDb();
            try {
            } finally {
                //执行完业务,要释放锁,使用delete解锁,非原子性,存在缺陷
                // 获取值对比+对比成功删除=原子操作
                // String lockValue = redisTemplate.opsForValue().get("lock");
                // if (uuid.equals(lockValue)) {
                //     redisTemplate.delete("lock");
                // }
                // 使用lua脚本解锁,保证原子性
                String script = "if redis.call('get',KEYS[1]) == ARGV[1] then return redis.call('del',KEYS[1]) else return 0 end";
                Long unlock = redisTemplate.execute(new DefaultRedisScript<Long>(script, Long.class), Arrays.asList("lock"), uuid);
            }
            return map;

        }else{
            //加锁失败,重试,执行类似syncronized方法,自旋锁,等待->重试
            System.out.println("获取分布式锁失败........等待重试");
            //可以添加休眠时间
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            return this.getCatelogJsonFromDbWithRedisLock();
        }

    }

2.2 使用Redisson 实现分布式锁:

Redisson 是架设在 Redis 基础上的一个 Java 驻内存数据网格(In-Memory Data Grid) 。 充分的利用了 Redis 键值数据库提供的一系列优势, 基于 Java 实用工具包中常用接口, 为使用者提供了一系列具有分布式特性的常用工具类。 使得原本作为协调单机多线程并发程序的工具包获得了协调分布式多机多线程并发系统的能力, 大大降低了设计和研发大规模分布式系统的难度。 同时结合各富特色的分布式服务, 更进一步简化了分布式环境中程序相互之间的协作。

官方文档: https://github.com/redisson/redisson/wiki/目录

使用方法:

1、maven引入依赖

复制代码
<!--使用redisson作为所有分布式锁,分布式对象等功能框架-->
 <dependency>
     <groupId>org.redisson</groupId>
     <artifactId>redisson</artifactId>
     <version>3.12.0</version>
 </dependency>

2、 配置:

java 复制代码
// 默认连接地址 127.0.0.1:6379

RedissonClient redisson = Redisson.create();

Config config = new Config(); //redis://127.0.0.1:7181

//可以用"rediss://"来启用 SSL 连接

config.useSingleServer().setAddress("redis://192.168.56.10:6379");

RedissonClient redisson = Redisson.create(config);

3、 使用分布式锁

java 复制代码
RLock lock = redisson.getLock("anyLock");// 最常见的使用方法
lock.lock();
// 加锁以后 10 秒钟自动解锁// 无需调用 unlock 方法手动解锁
lock.lock(10, TimeUnit.SECONDS);
// 尝试加锁, 最多等待 100 秒, 上锁以后 10 秒自动解锁
boolean res = lock.tryLock(100,10, TimeUnit.SECONDS);
if (res) {
try {
   
   ...

} finally {
   lock.unlock();
  }
}
相关推荐
天空属于哈夫克315 分钟前
Java 版:利用外部群 API 实现自动“技术开课”倒计时提醒
数据库·python·mysql
eWidget30 分钟前
随机森林原理:集成学习思想 —— Java 实现多棵决策树投票机制
java·数据库·随机森林·集成学习·金仓数据库
Traced back38 分钟前
SQL Server 核心语法+进阶知识点大全(小白版)
数据库·sqlserver
资深web全栈开发43 分钟前
PostgreSQL枚举还是字符串:ENUM vs VARCHAR + CHECK 的权衡
数据库·postgresql
为什么不问问神奇的海螺呢丶43 分钟前
n9e categraf rabbitmq监控配置
分布式·rabbitmq·ruby
凯子坚持 c1 小时前
C++基于微服务脚手架的视频点播系统---客户端(4)
数据库·c++·微服务
OceanBase数据库官方博客1 小时前
OceanBase场景解码系列三|OB Cloud 如何稳定支撑中企出海实现数 10 倍的高速增长?
数据库·oceanbase·分布式数据库
m0_561359671 小时前
使用Python处理计算机图形学(PIL/Pillow)
jvm·数据库·python
山岚的运维笔记1 小时前
SQL Server笔记 -- 第14章:CASE语句
数据库·笔记·sql·microsoft·sqlserver
Data_Journal1 小时前
如何使用 Python 解析 JSON 数据
大数据·开发语言·前端·数据库·人工智能·php