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 相关文章推荐
在同一窗体中使用PHP来处理多个提交任务
Oct 09 PHP
php 数组的创建、调用和更新实现代码
Mar 09 PHP
php 静态页面中显示动态内容
Aug 14 PHP
PHP IN_ARRAY 函数使用注意事项
Jul 24 PHP
apache+php完美解决301重定向的两种方法
Jun 08 PHP
PHP 导出Excel示例分享
Aug 18 PHP
php实现通用的信用卡验证类
Mar 24 PHP
详解php设置session(过期、失效、有效期)
Nov 12 PHP
PHP闭包函数详解
Feb 13 PHP
php判断手机浏览还是web浏览,并执行相应的动作简单实例
Jul 28 PHP
php进程daemon化的正确实现方法
Sep 06 PHP
php使用fputcsv实现大数据的导出操作详解
Feb 27 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&amp;&amp;mysql)六
2006/10/09 PHP
PHP clearstatcache()函数详解
2010/03/02 PHP
解析PHP中ob_start()函数的用法
2013/06/24 PHP
php检测数组长度函数sizeof与count用法
2014/11/17 PHP
Laravel接收前端ajax传来的数据的实例代码
2017/07/20 PHP
thinkphp中U方法按路由规则生成url的方法
2018/03/12 PHP
JavaScript实现页面滚动图片加载(仿lazyload效果)
2011/07/22 Javascript
javascript中的注释使用与注意事项小结
2011/09/20 Javascript
深入Javascript函数、递归与闭包(执行环境、变量对象与作用域链)使用详解
2013/05/08 Javascript
JavaScript定时显示广告代码分享
2015/03/02 Javascript
JavaScript驾驭网页-获取网页元素
2016/03/24 Javascript
Javascript之Math对象详解
2016/06/07 Javascript
vue.js初学入门教程(2)
2016/11/07 Javascript
JS针对Array的各种操作汇总
2016/11/29 Javascript
vue-cli中vue本地实现跨域调试接口
2019/01/16 Javascript
浅谈小程序 setData学问多
2019/02/20 Javascript
Vue中遍历数组的新方法实例详解
2019/07/21 Javascript
VsCode与Node.js知识点详解
2019/09/05 Javascript
Element实现表格嵌套、多个表格共用一个表头的方法
2020/05/09 Javascript
[56:20]LGD vs VP Supermajor 败者组决赛 BO3 第三场 6.10
2018/07/04 DOTA
简单介绍Python中利用生成器实现的并发编程
2015/05/04 Python
Python3计算三角形的面积代码
2017/12/18 Python
Pytorch 实现自定义参数层的例子
2019/08/17 Python
python代码xml转txt实例
2020/03/10 Python
使用Python构造hive insert语句说明
2020/06/06 Python
python类共享变量操作
2020/09/03 Python
python爬取招聘要求等信息实例
2020/11/20 Python
CSS3美化表单控件全集
2016/06/29 HTML / CSS
HTML5之WebGL 3D概述(上)—WebGL原生开发开启网页3D渲染新时代
2013/01/31 HTML / CSS
Java TransactionAPI (JTA) 主要包含几部分
2012/12/07 面试题
PyQt QMainWindow的使用示例
2021/03/24 Python
计算机软件个人的自荐信范文
2013/12/01 职场文书
迟到检讨书900字
2014/01/14 职场文书
《颐和园》教学反思
2014/02/26 职场文书
涉及车辆房产分割的离婚协议书范文
2014/10/12 职场文书
《伯牙绝弦》教学反思
2016/02/16 职场文书