【PHP】PHP+Redis+Lua 实现Redis分布式锁

PHP+Redis+Lua 实现Redis分布式锁

PHP部分代码:

<?php

class locks{
    private $redis;
    protected  $lockid;    //记录客户端加锁的ID

    public function __construct(){
        $redis = new Redis();
        $redis->connect('127.0.0.1', 6379);
        $this->redis = $redis;
    }

    /**
     * @param string $name 参数名
     * @param int $exp 过期时间
     * @param int $retry 重复次数
     * @param int $sleep  等待时间
     */
    public function lock($name = 'lock',$exp = 30, $retry = 10 , $sleep = 1){
        $result = false;
        while ($retry-- >= 0){
            $value  = md5(date("Y-m-d H:i:s"));
            $result = $this->redis->set( $name , $value , ['NX' , 'EX'=>$exp] );
            if ($result) {
                $this->lockid[$name] = $value;
                echo $name.' 成功获取到锁,正在执行' . "\n";
                break;
            }
            echo $name.'加锁失败·· 正常尝试获取锁' ."\n";
            sleep($sleep);
        }
        return $result;
    }

    /**
     * @param $name
     * @return mixed
     * 删除锁
     */
    public function unlock($name){
        echo $name.' 正常执行解锁···' ."\n";
        $lua = "
            local key   = KEYS[1]
            local value = ARGV[1]
            
            if(redis.call('get', key) == value)
            then
                return redis.call('del', key)
            end";

        return $this->redis->eval($lua, [$name, $this->lockid[$name]], 1);
    }

}

// 申请分布式锁
$obj = new locks();

// 执行完成申请解锁
if($obj->lock($name = "lock_1")){
    echo "执行结束,等待5s执行解锁操作···"."\n";
    sleep(5);
    echo "--------------------------"."\n";
    $unlockResult = $obj->unlock($name);
    if($unlockResult){
        echo $name.' 解锁成功··' ."\n";
    }
}

?>

Redis申请分布式锁的命令,此命令符合原子性:

 /**
     * SET:            Redis的String指令
     * lock_name:      锁的名称
     * unique_id:      锁的唯一校验
     * NX:             只有当lock_name的值不存在的时候才能SET成功
     * PX:             设置过期时间
*/

SET lock_name unique_id NX PX expire_time

开启两个窗口执行,分别执行申请锁的命令:

<?php
// 申请分布式锁
$obj = new locks();

// 执行完成申请解锁
if($obj->lock($name = "lock_1")){
    echo "执行结束,等待5s执行解锁操作···"."\n";
    sleep(5);
    echo "--------------------------"."\n";
    $unlockResult = $obj->unlock($name);
    if($unlockResult){
        echo $name.' 解锁成功··' ."\n";
    }
}

窗口1执行结果:
在这里插入图片描述
窗口2执行结果:
在这里插入图片描述

简简单单的使用PHP实现了Redis分布式锁

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值