Redis分布式锁解决超卖问题

简介: Redis分布式锁解决超卖问题

1、三种分布式锁实现方式对比:

1、数据库分布式锁实现缺点:

1)db操作性能较差,且有锁表的风险;

2)非阻塞操作失败后,需要轮询,占用cpu资源;

3)长时间不commit或者长时间轮询,可能会占用较多连接资源。

2、ZK分布式锁实现缺点:

性能不如redis,因为其写操作(获取锁释放锁)都需要在Leader上执行,然后同步到follower。

3、Redis(缓存)分布式锁实现缺点:

1)过期时间不好控制;

2)非阻塞,操作失败后,需要轮询,占用cpu资源;

2、使用原生redis实现

使用原生Redis的SetNX+Expire实现的分布式锁。

//方案1 setnx
String lockKey = "zhyRedis";
   //通过val,给锁设置唯一id,防止其他线程删除锁
String clientId = UUID.randomUUID().toString();  //或者雪花生成位置ID
boolean result = redisTemplate.opsForValue().setIfAbsent(lockKey, clientId, 10, TimeUnit.SECONDS); 
  // redisTemplate.expire(lockKey,10,TimeUnit.SECONDS) //旧版本redis(新版直接设置在后面,如上)
  //如果获取不到锁,则返回失败
if(!result){
    result "failed";
}
try {
 //如果能获取到锁,则返回成功
 Integer count = Integer.parseInt(redisTemplate.opsForValue.get("count").toString());
 if (count > 0) {
  Integer realCount = count - 1;
  System.out.Println("购买成功,剩余库存:" + realCount.toString());
  redisTemplate.opsForValue().set("count",realCount.toString());
 }else{
  System.out.Println("购买失败,库存不足");
 }
} catch(Exception e){
 e.printStackTrace();
  return "failed";
} finally{
  //解锁
  //判断当前客户端id与redis分布式中持有的客户端id一致,才能删除锁
 if(clientId.equals(redisTemplate.opsForValue().get(lockKey))){
   redisTemplate.delete(lockKey)
 }
}

3、使用redisson实现

相较于原生Redis的SetNX+Expire实现的分布式锁而言,Redisson的分布式锁组件可以解决原生Redis组合命令带来的一些缺陷,即redis的超时时间值不知道设置为多少才合适。如果此时Redis的服务器节点恰好出现宕机或者服务不能用的情况,那将会导致相应的Key永远存于缓存中,即处于所谓的“永久被锁死”的状态!

底层的实现机制在于:Redisson内部提供了一个监控锁的看门狗WatchDog,其作用在于Redis实例被关闭之前,不断延长锁的有效期。

除此之外,Redisson还通过加锁的方法提供了leaseTime等参数来指定加锁的有效时间,即超过这个时间后“锁”便自动解开了。

//方案2 redisson
String lockKey = "zhyRedis";
RLock rLock = redisson.getLock(lockKey);
try {
 rLock.lock(10, TimeUnit.SECONDS)
 //如果能获取到锁,则返回成功
 Integer count = Integer.parseInt(redisTemplate.opsForValue.get("count").toString());
 if (count > 0) {
  Integer realCount = count - 1;
  System.out.Println("购买成功,剩余库存:" + realCount.toString());
  redisTemplate.opsForValue().set("count",realCount.toString());
 }else{
  System.out.Println("购买失败,库存不足");
 }
} catch(Exception e){
 e.printStackTrace();
 return "failed";
} finally{
 //解锁
 rLock.unlock();
}
相关文章
|
9月前
|
存储 负载均衡 NoSQL
【赵渝强老师】Redis Cluster分布式集群
Redis Cluster是Redis的分布式存储解决方案,通过哈希槽(slot)实现数据分片,支持水平扩展,具备高可用性和负载均衡能力,适用于大规模数据场景。
600 2
|
9月前
|
存储 缓存 NoSQL
【📕分布式锁通关指南 12】源码剖析redisson如何利用Redis数据结构实现Semaphore和CountDownLatch
本文解析 Redisson 如何通过 Redis 实现分布式信号量(RSemaphore)与倒数闩(RCountDownLatch),利用 Lua 脚本与原子操作保障分布式环境下的同步控制,帮助开发者更好地理解其原理与应用。
734 6
|
10月前
|
存储 缓存 NoSQL
Redis核心数据结构与分布式锁实现详解
Redis 是高性能键值数据库,支持多种数据结构,如字符串、列表、集合、哈希、有序集合等,广泛用于缓存、消息队列和实时数据处理。本文详解其核心数据结构及分布式锁实现,帮助开发者提升系统性能与并发控制能力。
|
8月前
|
NoSQL Java 调度
分布式锁与分布式锁使用 Redis 和 Spring Boot 进行调度锁(不带 ShedLock)
分布式锁是分布式系统中用于同步多节点访问共享资源的机制,防止并发操作带来的冲突。本文介绍了基于Spring Boot和Redis实现分布式锁的技术方案,涵盖锁的获取与释放、Redis配置、服务调度及多实例运行等内容,通过Docker Compose搭建环境,验证了锁的有效性与互斥特性。
722 0
分布式锁与分布式锁使用 Redis 和 Spring Boot 进行调度锁(不带 ShedLock)
|
8月前
|
缓存 NoSQL 关系型数据库
Redis缓存和分布式锁
Redis 是一种高性能的键值存储系统,广泛用于缓存、消息队列和内存数据库。其典型应用包括缓解关系型数据库压力,通过缓存热点数据提高查询效率,支持高并发访问。此外,Redis 还可用于实现分布式锁,解决分布式系统中的资源竞争问题。文章还探讨了缓存的更新策略、缓存穿透与雪崩的解决方案,以及 Redlock 算法等关键技术。
|
10月前
|
NoSQL Redis
Lua脚本协助Redis分布式锁实现命令的原子性
利用Lua脚本确保Redis操作的原子性是分布式锁安全性的关键所在,可以大幅减少由于网络分区、客户端故障等导致的锁无法正确释放的情况,从而在分布式系统中保证数据操作的安全性和一致性。在将这些概念应用于生产环境前,建议深入理解Redis事务与Lua脚本的工作原理以及分布式锁的可能问题和解决方案。
347 8
|
NoSQL Redis 数据库
用redis实现分布式锁时容易踩的5个坑
云栖号资讯:【点击查看更多行业资讯】在这里您可以找到不同行业的第一手的上云资讯,还在等什么,快来! 近有不少小伙伴投入短视频赛道,也出现不少第三方数据商,为大家提供抖音爬虫数据。 小伙伴们有没有好奇过,这些数据是如何获取的,普通技术小白能否也拥有自己的抖音爬虫呢? 本文会全面解密抖音爬虫的幕后原理,不需要任何编程知识,还请耐心阅读。
用redis实现分布式锁时容易踩的5个坑
|
NoSQL Java 关系型数据库
浅谈Redis实现分布式锁
浅谈Redis实现分布式锁
|
存储 canal 缓存
|
NoSQL PHP Redis
redis实现分布式锁
redis实现分布式锁
354 0
redis实现分布式锁