详解如何实现一个 Redis 分布式锁

时间:2020-08-03 11:50:17 来源: 马哥Linux运维


在我们日常开发中,难免会遇到要加锁的情景。例如扣除产品库存,首先要从数据库中取出库存,进行库存判断,再减去库存。这一波操作明显不符合原子性,如果代码块不加锁,很容易因为并发导致超卖问题。咱们的系统如果是单体架构,那我们使用本地锁就可以解决问题。如果是分布式架构,就需要使用分布式锁。

方案

使用 SETNX 和 EXPIRE 命令

SETNXkeyvalue

EXPIREkeyseconds

DELkey

if(setnx("item_1_lock",1)){

expire("item_1_lock",30);

try{

...逻辑

}catch{

...

}finally{

del("item_1_lock");

}

}

这种方法看起来可以解决问题,但是有一定的风险,因为 SETNX 和 EXPIRE 这波操作是非原子性的,如果 SETNX 成功之后,出现错误,导致 EXPIRE 没有执行,导致锁没有设置超时时间形成死锁。

针对这种情况,我们可以使用 lua 脚本来保持操作原子性,保证 SETNX 和 EXPIRE 两个操作要么都成功,要么都不成功。

if(redis.call('setnx',KEYS[1],ARGV[1])<1)

thenreturn0;

end;

redis.call('expire',KEYS[1],tonumber(ARGV[2]));

return1;

通过这样的方法,我们初步解决了竞争锁的原子性问题,虽然其他功能还未实现,但是应该不会造成死锁

关键词:Redis 分布式锁

关于我们 加入我们 广告服务 网站地图

All Rights Reserved, Copyright 2004-2020 www.ctocio.com.cn

如有意见请与我们联系 邮箱:5 53 13 8 [email protected]

豫ICP备20005723号    IT专家网 版权所有