【Redis实战】分布式锁的N种实现方案对比与避坑指南在高并发场景下,分布式锁是保证数据一致性的关键技术。本文将从原理到实战,详细讲解分布式锁的各种实现方案。一、为什么需要分布式锁?假设这样一个场景:双十一秒杀活动,库存只剩1件,但有两个用户同时下单。如果没有锁机制,可能两个用户都抢到了这件商品,导致超卖问题。单机环境下,我们可以用 synchronized 或 ReentrantLock 来解决。但如果是分布式部署,多个服务实例同时抢锁,JVM锁就失效了------因为每个实例都有自己的锁对象,彼此之间完全感知不到。分布式锁的核心目标:互斥性:同一时刻只能有一个客户端持有锁防死锁:即使客户端崩溃,锁也要能自动释放可重入:同一个客户端可以多次获取锁高性能:加锁/解锁操作要快二、方案一:SET NX EX(最简单粗暴)这是很多人第一次用Redis实现分布式锁的方式:public Boolean tryLock(String key, String value, long expireTime) {
return redisTemplate.opsForValue()
.setIfAbsent(key, value, expireTime, TimeUnit.SECONDS);
}优点:简单,一行代码搞定缺点:可靠性差,value 无法唯一标识锁的持有者三、方案二:SET NX PX + 唯一value(推荐入门)改进版,每个锁带唯一标识,解锁时校验:// 加锁
String uuid = UUID.randomUUID().toString();
Boolean success = redisTemplate.opsForValue()
.setIfAbsent(key, uuid, 30, TimeUnit.SECONDS);
// 解锁(Lua脚本保证原子性)
String script =
"if redis.call('get', KEYS[1]) == ARGV[1] then " +
" return redis.call('del', KEYS[1]) " +
else return 0 end";
redisTemplate.execute(
new DefaultRedisScript<>(script, Long.class),
Collections.singletonList(key), uuid
);四、方案三:Redisson(生产级方案)Redisson是Redis的Java客户端,封装了完善的分布式锁实现:@Autowired
private RedissonClient redissonClient;
public void businessMethod() {
RLock lock = redissonClient.getLock("myLock");
try {
// 等待30秒,锁定后自动续期
if (lock.tryLock(30, -1, TimeUnit.SECONDS)) {
// 业务逻辑
}
} finally {
if (lock.isHeldByCurrentThread()) {
lock.unlock();
}
}
}核心特性:自动续期:看门狗机制防止锁提前释放可重入:同一线程可多次获取锁公平锁/读写锁:丰富的锁类型支持五、避坑指南坑1:锁过期了但业务还没执行完问题:设置了30秒过期,但业务需要45秒解决方案:Redisson的看门狗机制会每10秒自动续期30秒坑2:主从切换导致锁丢失问题:Redis主节点加锁成功,但数据还没同步到从节点,主节点挂了解决方案:使用RedLock算法(多节点加锁)或使用Redisson Pro(支持Redis Cluster)坑3:解锁时误删了他人的锁问题:没有校验value,直接del了解决方案:解锁时用Lua脚本先判断value再删除六、总结
方案可靠性复杂度适用场景SET NX⭐⭐测试/低并发SET NX + value⭐⭐⭐⭐一般生产环境Redisson⭐⭐⭐⭐⭐⭐⭐高并发生产环境我的建议:直接上Redisson,省心省力。当然,理解底层原理也很重要,面试时能手写分布式锁会让你加分不少。相关阅读:Redis缓存实战:从入门到精通Spring Boot集成Redisson详细教程觉得有用的话,点个赞再走呗~ 有问题欢迎评论区交流!