资讯专栏INFORMATION COLUMN

php redis 加锁与解锁

JellyBool / 3280人阅读

摘要:将用户添加进房间李四王五李四王五张三脚本解锁先判断的值是否为才会删除所以的设计要有随机唯一性具体还可以看看这篇文章解锁锁的正确姿势还有操作的文档里面有函数的解释注意用脚本这里需要开放等系统函数以上代码仅作参考

php+redis 实现加锁与解锁操作

业务背景:在房间棋牌游戏中需要用到锁来防止并发操作引起的 redis 数据脏读问题;例如添加用户进入房间的动作:

并发的情况下,get RoomUsers 会有脏读现象;


解决思路:加锁房间来实现 一个房间每次只允许一个客户端操作,其他并发客户端则等待;也就是-----堵塞锁;


加锁:redis加锁方式有几种: incr、set、setnx、hSetnx,可以参考这篇文章:redis加锁的几种实现

这里我用到 set 这种方式

$roomId = $_GET["roomId"];
$user = $_GET["user"];             // "张三"
$key = "LockRoom:{$roomId}";
$value = $roomId.uniqid();
$ex = 3;
// 如果 $key 不存在的话,就设置 $key 的值为 $value,且有效期为 3s; 
// return TRUE / FALSE
while(true){
    $res  = $this->redis->set($key, $value, ["nx", "ex" => $ex]);
    if($res) { break; }
    usleep(5000);
}

// 将用户添加进房间
$roomUsers = $this->redis->get("Room:{$roomId}:Users"); // ["李四", "王五"]
$roomUsers[] = $user;
$this->redis->set("Room:{$roomId}:Users", $roomUsers); // ["李四", "王五", "张三"]

解锁:操作完当然要解锁了,不解锁起码要等待 3秒;
解锁用 delete 删除 key; 但是这里有个坑,不能直接用 delete,因为假设 client01 获得了锁,在添加用户进入房间的过程中 时间超过了 3秒 ,这个时候client02 就会同样获得锁并且设置3S,然后当client01 操作完之后 delete key , 就把 client02 设置的锁删除了;
这里推荐用 lua 代码执行删除,因为lua 执行具有原子性。

// 将用户添加进房间
$roomUsers = $this->redis->get("Room:{$roomId}:Users"); // ["李四", "王五"]
$roomUsers[] = $user;
$this->redis->set("Room:{$roomId}:Users", $roomUsers); // ["李四", "王五", "张三"]

// lua 脚本解锁
// 先判断 key的值是否为 value, TRUE 才会删除, 所以 $value 的设计要有随机唯一性
$script = "if redis.call("get",KEYS[1]) == ARGV[1]
then
    return redis.call("del",KEYS[1])
else
    return 0
end ";
$this->redis->eval($script, array($key , $value), 1);

具体还可以看看 这篇文章:解锁 Redis 锁的正确姿势

还有php操作redis的文档:PhpRedis 里面有 set()、eval() 函数的解释

注意:用 lua 脚本这里 php.ini 需要开放 shell_exec() 等系统函数
以上代码仅作参考!!

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/28856.html

相关文章

  • php redis 加锁解锁

    摘要:将用户添加进房间李四王五李四王五张三脚本解锁先判断的值是否为才会删除所以的设计要有随机唯一性具体还可以看看这篇文章解锁锁的正确姿势还有操作的文档里面有函数的解释注意用脚本这里需要开放等系统函数以上代码仅作参考 php+redis 实现加锁与解锁操作 业务背景:在房间棋牌游戏中需要用到锁来防止并发操作引起的 redis 数据脏读问题;例如添加用户进入房间的动作: showImg(http...

    K_B_Z 评论0 收藏0
  • 基于Redis实现分布式锁

    摘要:本篇博客将介绍第二种方式,基于的实现分布式锁。总结本文主要介绍了如何使用代码正确实现分布式锁,对于加锁和解锁也分别给出了两个比较经典的错误示例。其实想要通过实现分布式锁并不难,只要保证能满足可靠性里的四个条件。 前言 分布式锁一般有三种实现方式:1.数据库乐观锁;2、基于Redis的分布式锁;3.基于Zookeeper的分布式锁。本篇博客将介绍第二种方式,基于Redis的实现分布式锁。...

    jonh_felix 评论0 收藏0
  • 基于Redis实现分布式锁

    摘要:本篇博客将介绍第二种方式,基于的实现分布式锁。总结本文主要介绍了如何使用代码正确实现分布式锁,对于加锁和解锁也分别给出了两个比较经典的错误示例。其实想要通过实现分布式锁并不难,只要保证能满足可靠性里的四个条件。 前言 分布式锁一般有三种实现方式:1.数据库乐观锁;2、基于Redis的分布式锁;3.基于Zookeeper的分布式锁。本篇博客将介绍第二种方式,基于Redis的实现分布式锁。...

    李涛 评论0 收藏0
  • 【Java猫说】Java多线程之内存可见性(上篇)

    摘要:猫说多线程之内存可见性下篇欢迎你留言讨论属于你的见解,毕竟每个人的味蕾都不一样,这杯咖啡有吸引到你吗好像又是一个槽糕的比喻本文已转载个人技术公众号欢迎留言讨论与点赞上一篇推荐猫说主数据类型和引用下一篇推荐猫说多线程之内存可见性下篇 阅读本文约3分钟 本文大致讲述两种线程实现的可见性,或许你已经提前想到了,那说明你的基础很好,我们要聊聊synchronized实现可见性与volatil...

    khlbat 评论0 收藏0
  • 【Java猫说】Java多线程之内存可见性(上篇)

    摘要:猫说多线程之内存可见性下篇欢迎你留言讨论属于你的见解,毕竟每个人的味蕾都不一样,这杯咖啡有吸引到你吗好像又是一个槽糕的比喻本文已转载个人技术公众号欢迎留言讨论与点赞上一篇推荐猫说主数据类型和引用下一篇推荐猫说多线程之内存可见性下篇 阅读本文约3分钟 本文大致讲述两种线程实现的可见性,或许你已经提前想到了,那说明你的基础很好,我们要聊聊synchronized实现可见性与volatil...

    Heier 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<