php 使用redis锁限制并发访问类示例


Posted in PHP onNovember 02, 2016

本文介绍了php 使用redis锁限制并发访问类,并详细的介绍了并发访问限制方法。

1.并发访问限制问题

对于一些需要限制同一个用户并发访问的场景,如果用户并发请求多次,而服务器处理没有加锁限制,用户则可以多次请求成功。

例如换领优惠券,如果用户同一时间并发提交换领码,在没有加锁限制的情况下,用户则可以使用同一个换领码同时兑换到多张优惠券。

伪代码如下:

if A(可以换领)
    B(执行换领)
    C(更新为已换领)
D(结束)

如果用户并发提交换领码,都能通过可以换领(A)的判断,因为必须有一个执行换领(B)后,才会更新为已换领(C)。因此如果用户在有一个更新为已换领之前,有多少次请求,这些请求都可以执行成功。

2.并发访问限制方法

使用文件锁可以实现并发访问限制,但对于分布式架构的环境,使用文件锁不能保证多台服务器的并发访问限制。

Redis是一个开源的使用ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库,并提供多种语言的API。

本文将使用其setnx方法实现分布式锁功能。setnx即Set it N**ot eX**ists。

当键值不存在时,插入成功(获取锁成功),如果键值已经存在,则插入失败(获取锁失败)

RedisLock.class.PHP

<?php
/**
 * Redis锁操作类
 * Date:  2016-06-30
 * Author: fdipzone
 * Ver:  1.0
 *
 * Func:
 * public lock  获取锁
 * public unlock 释放锁
 * private connect 连接
 */
class RedisLock { // class start

  private $_config;
  private $_redis;

  /**
   * 初始化
   * @param Array $config redis连接设定
   */
  public function __construct($config=array()){
    $this->_config = $config;
    $this->_redis = $this->connect();
  }

  /**
   * 获取锁
   * @param String $key  锁标识
   * @param Int   $expire 锁过期时间
   * @return Boolean
   */
  public function lock($key, $expire=5){
    $is_lock = $this->_redis->setnx($key, time()+$expire);

    // 不能获取锁
    if(!$is_lock){

      // 判断锁是否过期
      $lock_time = $this->_redis->get($key);

      // 锁已过期,删除锁,重新获取
      if(time()>$lock_time){
        $this->unlock($key);
        $is_lock = $this->_redis->setnx($key, time()+$expire);
      }
    }

    return $is_lock? true : false;
  }

  /**
   * 释放锁
   * @param String $key 锁标识
   * @return Boolean
   */
  public function unlock($key){
    return $this->_redis->del($key);
  }

  /**
   * 创建redis连接
   * @return Link
   */
  private function connect(){
    try{
      $redis = new Redis();
      $redis->connect($this->_config['host'],$this->_config['port'],$this->_config['timeout'],$this->_config['reserved'],$this->_config['retry_interval']);
      if(empty($this->_config['auth'])){
        $redis->auth($this->_config['auth']);
      }
      $redis->select($this->_config['index']);
    }catch(RedisException $e){
      throw new Exception($e->getMessage());
      return false;
    }
    return $redis;
  }

} // class end

?>

demo.php

<?php
require 'RedisLock.class.php';

$config = array(
  'host' => 'localhost',
  'port' => 6379,
  'index' => 0,
  'auth' => '',
  'timeout' => 1,
  'reserved' => NULL,
  'retry_interval' => 100,
);

// 创建redislock对象
$oRedisLock = new RedisLock($config);

// 定义锁标识
$key = 'mylock';

// 获取锁
$is_lock = $oRedisLock->lock($key, 10);

if($is_lock){
  echo 'get lock success<br>';
  echo 'do sth..<br>';
  sleep(5);
  echo 'success<br>';
  $oRedisLock->unlock($key);

// 获取锁失败
}else{
  echo 'request too frequently<br>';
}

?>

测试方法:

打开两个不同的浏览器,同时在A,B中访问demo.php

如果先访问的会获取到锁

输出

get lock success
do sth..
success

另一个获取锁失败则会输出request too frequently

保证同一时间只有一个访问有效,有效限制并发访问。

为了避免系统突然出错导致死锁,所以在获取锁的时候增加一个过期时间,如果已超过过期时间,即使是锁定状态都会释放锁,避免死锁导致的问题。
源码下载地址:点击查看

PHP 相关文章推荐
dedecms系统常用术语汇总
Apr 03 PHP
php preg_match_all结合str_replace替换内容中所有img
Oct 11 PHP
基于empty函数的判断详解
Jun 17 PHP
解析PHP跳出循环的方法以及continue、break、exit的区别介绍
Jul 01 PHP
JoshChen_web格式编码UTF8-无BOM的小细节分析
Aug 16 PHP
Laravel 5框架学习之环境与配置
Apr 08 PHP
Yii2框架使用计划任务的方法
May 25 PHP
thinkPHP统计排行与分页显示功能示例
Dec 02 PHP
PHP实现的简单AES加密解密算法实例
May 29 PHP
浅析php如何实现爬取数据原理
Sep 27 PHP
PHP常用工具函数小结【移除XSS攻击、UTF8与GBK编码转换等】
Apr 27 PHP
php实现对短信验证码发送次数的限制实例讲解
Mar 04 PHP
Android AsyncTack 异步任务实例详解
Nov 02 #PHP
php array_pop 删除数组最后一个元素实例
Nov 02 #PHP
PHP设置images目录不充许http访问的方法
Nov 01 #PHP
PHP递归获取目录内所有文件的实现方法
Nov 01 #PHP
php获得文件夹下所有文件的递归算法的简单实例
Nov 01 #PHP
ecshop适应在PHP7的修改方法解决报错的实现
Nov 01 #PHP
遍历echsop的region表形成缓存的程序实例代码
Nov 01 #PHP
You might like
php中使用addslashes函数报错问题的解决方法
2013/02/06 PHP
163的邮件用phpmailer发送(实例详解)
2013/06/24 PHP
PHP闭包函数详解
2016/02/13 PHP
Symfony2之session与cookie用法小结
2016/03/18 PHP
thinkPHP利用ajax异步上传图片并显示、删除的示例
2018/09/26 PHP
PHP的重载使用魔术方法代码实例详解
2021/02/26 PHP
safari,opera嵌入iframe页面cookie读取问题解决方法
2010/06/23 Javascript
JQuery事件e参数的方法preventDefault()取消默认行为
2013/09/26 Javascript
js采用map取到id集合组并且实现点击一行选中一行
2013/12/16 Javascript
js中继承的几种用法总结(apply,call,prototype)
2013/12/26 Javascript
Node.js与PHP、Python的字符处理性能对比
2014/07/06 Javascript
js中将String转换为number以便比较
2014/07/08 Javascript
jQuery CSS()方法改变现有的CSS样式表
2014/09/09 Javascript
基于jquery ui的alert,confirm方案(支持换肤)
2015/04/03 Javascript
JavaScript验证Email(3种方法)
2015/09/21 Javascript
jQuery的文档处理程序详解
2016/05/10 Javascript
Vue 2.x教程之基础API
2017/03/06 Javascript
nodejs中art-template模板语法的引入及冲突解决方案
2017/11/07 NodeJs
微信小程序制作扭蛋机代码实例
2019/09/24 Javascript
使用JS location实现搜索框历史记录功能
2019/12/23 Javascript
使用python BeautifulSoup库抓取58手机维修信息
2013/11/21 Python
Python过滤函数filter()使用自定义函数过滤序列实例
2014/08/26 Python
编写Python脚本使得web页面上的代码高亮显示
2015/04/24 Python
编写Python脚本批量下载DesktopNexus壁纸的教程
2015/05/06 Python
python中json格式数据输出的简单实现方法
2016/10/31 Python
Android 兼容性问题:java.lang.UnsupportedOperationException解决办法
2017/03/19 Python
基于python 二维数组及画图的实例详解
2018/04/03 Python
使用pip安装python库的多种方式
2019/07/31 Python
Django中modelform组件实例用法总结
2020/02/10 Python
Python列表倒序输出及其效率详解
2020/03/04 Python
解决django migrate报错ORA-02000: missing ALWAYS keyword
2020/07/02 Python
python正则表达式 匹配反斜杠的操作方法
2020/08/07 Python
婚内分居协议书范文
2014/11/26 职场文书
2015年读书月活动总结
2015/03/26 职场文书
2016年国庆节假期旅游工作总结
2016/04/01 职场文书
css3 利用transform-origin 实现圆点分布在大圆上布局及旋转特效
2021/04/29 HTML / CSS