<?php
class ProcessRedisLock
{
/**
* redis key 前缀
*/
const KEY_PREFIX = 'PROCESS_REDIS_LOCK:'; /**
* 默认超时时间(秒)
*/
const DEFAULT_TIMEOUT = 5; /**
* 最大超时时间(秒)
*/
const MAX_TIMEOUT_SETTING = 60; /**
* 随机数的最小值
*/
const MIN_RAND_NUM = 0; /**
* 随机数的最大值
*/
const RAND_MAX_NUM = 100000; /**
* 每次取锁间隔毫秒数
*/
const GET_LOCK_SLEEP_MICRO_SECONDS = 0.1; /**
* @var mixed redis 实例
*/
private $redisIns; /**
* @var string 锁名
*/
private $lockName; /**
* @var int 超时时间,不可超过 self::MAX_TIMEOUT_SETTING
*/
private $timeout; /*
单元测试步骤
--------------------------------------------------------------------------------------
1.分别用一个chrome和一个ie,模拟并发请求:
1)http://xxx/test?queryName=task1&handlerTime=10&lockName=myLocktest&timeout=10
2)http://xxx/test?queryName=task2&handlerTime=10&lockName=myLocktest&timeout=5
2.如上
task1处理耗时10s,取锁超时时间10s
task2处理耗时10s,取锁超时时间5s
3.日志结果
2019-11-22 17:16:30 [task1]: 尝试取锁,超时时间设定10秒
2019-11-22 17:16:30 [task1]: 获取到锁,唯一标志:PROCESS_REDIS_LOCK:5dd7a76e1bafe37915
2019-11-22 17:16:32 [task2]: 尝试取锁,超时时间设定5秒
2019-11-22 17:16:35 [task1]: 释放锁
2019-11-22 17:16:35 [task2]: 获取到锁,唯一标志:PROCESS_REDIS_LOCK:5dd7a770d0c2b41355
2019-11-22 17:16:45 [task2]: 释放锁
-------------------------------------------------------------------------------------- 单元测试代码
--------------------------------------------------------------------------------------
// 浏览器模拟并发请求
// 注意:这里测试的时候如果session以文件存储的话,要避免两个窗口共用一个会话id,因为同样的会话id在session_start()时会锁文件
// 这样会造成请求阻塞,模拟不了请求并发的情况,所以应该使用两个不同的浏览器(如一个chrome,一个firefox)(同样的浏览器共享cookie也会导致拿到同样的会话id)
public function testAction()
{
$params = $this->getRequest()->getParams();
$this->requestTask($params['queryName'], $params['handlerTime'], $params['lockName'], $params['timeout']);
} // 模拟取锁、耗时操作、释放锁
public function requestTask($queryName, $handlerTime, $lockName, $timeout)
{
try {
// 获取redis实例
$processRedisLock = new ProcessRedisLock(redis(), $lockName, $timeout); $this->echoAndSaveInfo($queryName, "尝试取锁,超时时间设定{$timeout}秒"); // 取锁
$id = $processRedisLock->lock();
// 如果到了超时时间还未取到会返回false,则直接抛异常
if($id === false){
throw new Exception('获取锁失败');
} $this->echoAndSaveInfo($queryName, "获取到锁,唯一标志:".$id); // 模拟耗时操作
sleep($handlerTime); // 释放锁
$processRedisLock->unlock($id);
$this->echoAndSaveInfo($queryName, "释放锁");
} catch (Exception $e) {
$this->echoAndSaveInfo($queryName, $e->getMessage());
// do something
}
} // 输出并且记录日志
public function echoAndSaveInfo($queryName, $content)
{
$info = date('Y-m-d H:i:s') . " [{$queryName}]: {$content}" . PHP_EOL;
echo $info;
file_put_contents('test.txt', $info . PHP_EOL, FILE_APPEND);
}
--------------------------------------------------------------------------------------
*/ /**
* ProcessRedisLock constructor.
* @param $redisIns
* @param $lockName
* @param int $timeout
* @throws Exception
*/
public function __construct($redisIns, $lockName, $timeout = self::DEFAULT_TIMEOUT)
{
if (!$redisIns) {
new Exception('The redis instance is empty');
} if(!$lockName){
throw new Exception('Lock name invalid');
} // 校验超时时间
$timeout = intval($timeout);
if (!($timeout > 0 && $timeout <= self::MAX_TIMEOUT_SETTING)) {
throw new Exception('The timeout interval is (0,' . self::MAX_TIMEOUT_SETTING . ']');
} $this->redisIns = $redisIns;
$this->lockName = $lockName;
$this->timeout = $timeout;
} /**
* 加锁
* @return bool
* @Date 2019/11/22
*/
public function lock()
{
// redis key
$key = $this->getRedisKey(); // 唯一标志
$id = $this->getId(); // 超时时间
$endTime = time() + $this->timeout; // 循环取锁
while (time() < $endTime) {
// 尝试加锁,若给定的 key 已经存在,则 SETNX 不做任何动作。
if ($this->redisIns->setnx($key, $id)) {
// 设置过期时间,防止程序异常退出没有解锁导致死锁
$this->redisIns->expire($key, $this->timeout);
// 返回唯一标志,用于解锁
return $id;
}
usleep(self::GET_LOCK_SLEEP_MICRO_SECONDS);
} return false;
} /**
* 解锁
* @param string $id 唯一标志,加锁成功时返回
* @return bool
* @Date 2019/11/22
*/
public function unlock($id)
{
$key = self::getRedisKey(); // 如果锁的值与没有被修改
if ($this->redisIns->get($key) == $id) {
// 开始事务
$this->redisIns->multi();
// 释放该锁
$this->redisIns->del($key);
// 执行
$this->redisIns->exec();
return true;
} else {
return false;
}
} /**
* 获取redis key
* @return string
* @Date 2019/11/22
*/
public function getRedisKey()
{
return self::KEY_PREFIX . $this->lockName;
} /**
* 获取唯一标志位
* @Date 2019/11/22
*/
public function getId()
{
return uniqid(self::KEY_PREFIX) . mt_rand(self::MIN_RAND_NUM, self::RAND_MAX_NUM);
}
}

PHP 基于redis的分布式锁的更多相关文章

  1. 基于redis 实现分布式锁的方案

    在电商项目中,经常有秒杀这样的活动促销,在并发访问下,很容易出现上述问题.如果在库存操作上,加锁就可以避免库存卖超的问题.分布式锁使分布式系统之间同步访问共享资源的一种方式 基于redis实现分布式锁 ...

  2. 基于redis的分布式锁

    <?php /** * 基于redis的分布式锁 * * 参考开源代码: * http://nleach.com/post/31299575840/redis-mutex-in-php * * ...

  3. 基于Redis的分布式锁真的安全吗?

    说明: 我前段时间写了一篇用consul实现分布式锁,感觉理解的也不是很好,直到我看到了这2篇写分布式锁的讨论,真的是很佩服作者严谨的态度, 把这种分布式锁研究的这么透彻,作者这种技术态度真的值得我好 ...

  4. 基于 Redis 的分布式锁

    前言 分布式锁在分布式应用中应用广泛,想要搞懂一个新事物首先得了解它的由来,这样才能更加的理解甚至可以举一反三. 首先谈到分布式锁自然也就联想到分布式应用. 在我们将应用拆分为分布式应用之前的单机系统 ...

  5. 基于redis的分布式锁(转)

    基于redis的分布式锁 1 介绍 这篇博文讲介绍如何一步步构建一个基于Redis的分布式锁.会从最原始的版本开始,然后根据问题进行调整,最后完成一个较为合理的分布式锁. 本篇文章会将分布式锁的实现分 ...

  6. 基于redis的分布式锁实现

    1.分布式锁介绍 在计算机系统中,锁作为一种控制并发的机制无处不在. 单机环境下,操作系统能够在进程或线程之间通过本地的锁来控制并发程序的行为.而在如今的大型复杂系统中,通常采用的是分布式架构提供服务 ...

  7. 基于redis的分布式锁(不适合用于生产环境)

    基于redis的分布式锁 1 介绍 这篇博文讲介绍如何一步步构建一个基于Redis的分布式锁.会从最原始的版本开始,然后根据问题进行调整,最后完成一个较为合理的分布式锁. 本篇文章会将分布式锁的实现分 ...

  8. 基于 redis 的分布式锁实现 Distributed locks with Redis debug 排查错误

    小结: 1. 锁的实现方式,按照应用的实现架构,可能会有以下几种类型: 如果处理程序是单进程多线程的,在 python下,就可以使用 threading 模块的 Lock 对象来限制对共享变量的同步访 ...

  9. 转载:基于Redis实现分布式锁

    转载:基于Redis实现分布式锁  ,出处: http://blog.csdn.net/ugg/article/details/41894947 背景在很多互联网产品应用中,有些场景需要加锁处理,比如 ...

  10. redis系列:基于redis的分布式锁

    一.介绍 这篇博文讲介绍如何一步步构建一个基于Redis的分布式锁.会从最原始的版本开始,然后根据问题进行调整,最后完成一个较为合理的分布式锁. 本篇文章会将分布式锁的实现分为两部分,一个是单机环境, ...

随机推荐

  1. 使用 Visual Studio Code(VSCode)搭建简单的Python+Django开发环境的方法步骤

    安装配置 VSCode [1]安装 VSCode: 下载地址:https://code.visualstudio.com/ 根据自己电脑对应的操作系统下载对应的版本即可,至于安装过程也和一般的软件一样 ...

  2. vux中XDialog组件,修改weui-mask(半透明遮罩)

    场景:XDialog组件的遮罩的透明度是background: rgba(0, 0, 0, 0.6);而期望是75%的透明度: 项目构成:vue,vux 最终实现效果: 思路:首先想到了修改组件下we ...

  3. C++内存管理4-Windows编程中的堆管理(转)

    1 引言 在大多数Windows应用程序设计中,都几乎不可避免的要对内存进行操作和管理.在进行大尺寸内存的动态分配时尤其显的重要.本文即主要对内存管理中的堆管理技术进行论述. 堆(Heap)实际是位于 ...

  4. 文件组 'PRIMARY' 已满 解决办法

    修改一个字段类型时,报的这个错. 此时需要增加次要数据文件 次要数据文件 次要数据文件包含除主要数据文件外的所有数据文件.有些数据库可能没有次要数据文件,而有些数据库则有多个次要数据文件.次要数据文件 ...

  5. 转载: 我如何使用 Django + Vue.js 快速构建项目

    原文链接: https://www.ctolib.com/topics-109796.html 正文引用如下 引言 大U的技术课堂 的新年第一课,祝大家新的一年好好学习,天天向上:) 本篇将手把手教你 ...

  6. IDEA 2018.3.5,修改js文件,html页面不及时更新

    问题描述 使用IDEA 开发时,修改js文件,前端页面不能及时更新. 解决方法: 1. IDEA settings--> Compiler --> Build project automa ...

  7. spring boot 2X中@Scheduled实现定时任务及多线程配置

    使用@Scheduled 可以很容易实现定时任务 spring boot的版本 2.1.6.RELEASE package com.abc.demo.common; import org.slf4j. ...

  8. 【windows】win10新增用户

    1.打开[计算机管理]

  9. 【sqlserver】【mysql】用NavicatPremium12把sqlserver的数据迁移到mysql上

    1.在NavicatPremium的mysql,选择到迁移到的表,然后右键[导入向导]

  10. pom中parent和dependency区别以及dependencyManagement区别

    真的很详细 很感动 1.在同一个pom文件下,如果<dependencies>和<dependencyManagement>中都对该jar做了依赖,以<dependenc ...