nx set 怎么实现的原子性_Redis 使用 Lua 脚本替代 SETNX / DECR 保证原子性

背景

最近公司出了一起故障,问题代码如下:

/**

* TRUE: 触发限流,FALSE:未触发限流

*/

public function acquire() {

try {

$redisHandler = $this->redisInstance->getHandler();

$redisHandler->set($this->rateLimitKey, $this->tokenNum, ['nx', 'ex' => $this->expireTime]);

$leftTokenNum = $redisHandler->decr($this->rateLimitKey);

if ($leftTokenNum < 0) {

return TRUE;

}

return FALSE;

} catch (\Exception $e) {

return FALSE;

}

}

作者的目的是针对爆款商品的购买,使用 redis 来起到一个限流的作用,1 秒钟只允许 1 人购买。

结果上线过后不久,运营就反馈线上出故障了,该爆款商品所有人都不能购买了。

分析

上面代码的思路很简单:通过 $redis->set('key', '1', ['nx', 'ex'=>1]); 命令,设置值为 1 过期时间为 1 秒的计数器,基于该计数器的

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值