Redis 分散式鎖解決方案

vasar發表於2018-08-14

在論壇上找到各種版本,應用到專案出執行不了,各種報錯。今天自己重新編寫除錯測試,執行成功。

  • PHP redis使用phpredis
  • 參考redis官方資料
  • 具體程式碼

    <?php
    namespace app\common\service;
    
    /**
     *
     * redis 加鎖 --單Redis例項實現分散式鎖
     *
     * -- 分散式請使用:Redlock:https://github.com/ronnylt/redlock-php
     * -- 詳情參考: http://www.redis.cn/topics/distlock.html
     *
     * @package app\common\service
     */
    class RedisLock
    {
        const LOCK_SUCCESS = 'OK';
        const IF_NOT_EXISTS = 'NX';
        const MILLISECOND_EXPIRE_TIME = 'PX';
        const EXPIRE_TIME = 60000; // millisecond => 60s
        const LOCK_VALUE = 1;
    
            /**
             * 加鎖
             * @param $redis object
             * @param $key
             * @param string $expire_time 60000
             */
            public static function lock($redis, $key, $expire_time='')
            {
                if (empty($expire_time)) {
                        $expire_time = self::EXPIRE_TIME;
                }
    
                $result = $redis->set($key, self::LOCK_VALUE, [self::IF_NOT_EXISTS, self::MILLISECOND_EXPIRE_TIME => $expire_time]);
    
                return self::LOCK_SUCCESS === (string)$result;
            }
    
            /**
             * 解鎖
             *
             * 參考: https://github.com/phpredis/phpredis/blob/develop/tests/RedisTest.php
             * @param $redis
             * @param $key
             */
            public static function unlock($redis, $key)
            {
                $lua =<<<EOT
    if redis.call("get",KEYS[1]) == ARGV[1] then
        return redis.call("del",KEYS[1])
    else
        return 0
    end
    EOT;
                $result = $redis->eval($lua, array($key, self::LOCK_VALUE), 1);
                return $result;
        }
    }
本作品採用《CC 協議》,轉載必須註明作者和本文連結
欲速則不達

相關文章