PHP 并发场景的几种解决方案


Posted in PHP onJune 14, 2019

在秒杀,抢购等并发场景下,可能会出现超卖的现象,在PHP语言中并没有原生提供并发的解决方案,因此就需要借助其他方式来实现并发控制。

列出常见的解决方案有:

  • 使用队列,额外起一个进程处理队列,并发请求都放到队列中,由额外进程串行处理,并发问题就不存在了,但是要额外进程支持以及处理延迟严重,本文不先不讨论这种方法。
  • 利用数据库事务特征,做原子更新,此方法需要依赖数据库的事务特性。
  • 借助文件排他锁,在处理下单请求的时候,用flock锁定一个文件,成功拿到锁的才能处理订单。

一、利用 Redis 事务特征

redis 事务是原子操作,可以保证订单处理的过程中数据没有被其它并发的进程修改。

示例代码:

<?php
$http = new swoole_http_server("0.0.0.0", 9509);  // 监听 9509

$http->set(array(
  'reactor_num' => 2, //reactor thread num
  'worker_num' => 4  //worker process num
));

$http->on('request', function (swoole_http_request $request, swoole_http_response $response) {
  $uniqid = uniqid('uid-', TRUE);  // 模拟唯一用户ID
  $redis = new Redis();
  $redis->connect('127.0.0.1', 6379);  // 连接 redis

  $redis->watch('rest_count'); // 监测 rest_count 是否被其它的进程更改

  $rest_count = intval($redis->get("rest_count")); // 模拟唯一订单ID
  if($rest_count > 0){
    $value = "{$rest_count}-{$uniqid}"; // 表示当前订单,被当前用户抢到了

    // do something ... 主要是模拟用户抢到单后可能要进行的一些密集运算
    $rand = rand(100, 1000000);
    $sum=0;
    for ($i=0;$i<$rand;$i++){ $sum+=$i; }

   // redis 事务
    $redis->multi();
    $redis->lPush('uniqids', $value);
    $redis->decr('rest_count');
    $replies = $redis->exec(); // 执行以上 redis 事务

   // 如果 rest_count 的值被其它的并发进程更改了,以上事务将回滚
    if(!$replies){
      echo "订单 {$value} 回滚".PHP_EOL;
    }
  }
  $redis->unwatch();
});

$http->start();

使用 ab 测试

$ ab -t 20 -c 10 http://192.168.1.104:9509/

二、利用文件排他锁(阻塞模式)

阻塞模式下,如果进程在获取文件排他锁时,其它进程正在占用锁的话,此进程会挂起等待其它进程释放锁后,并自己获取到锁后,再往下执行。

示例代码:

<?php
$http = new swoole_http_server("0.0.0.0", 9510);

$http->set(array(
  'reactor_num' => 2, //reactor thread num
  'worker_num' => 4  //worker process num
));

$http->on('request', function (swoole_http_request $request, swoole_http_response $response) {

  $uniqid = uniqid('uid-', TRUE);
  $redis = new Redis();
  $redis->connect('127.0.0.1', 6379);

  $fp = fopen("lock.txt", "w+");

  // 阻塞(等待)模式, 要取得独占锁定(写入的程序)
  if(flock($fp,LOCK_EX))  //锁定当前指针
  {
   // 成功取得锁后,放心处理订单
    $rest_count = intval($redis->get("rest_count"));
    $value = "{$rest_count}-{$uniqid}";
    if($rest_count > 0){
      // do something ...
      $rand = rand(100, 1000000);
      $sum=0;
      for ($i=0;$i<$rand;$i++){ $sum+=$i; }

      $redis->lPush('uniqids', $value);
      $redis->decr('rest_count');
    }

   // 订单处理完成后,再释放锁
    flock($fp,LOCK_UN);
  }
  fclose($fp);

});

$http->start();

使用 ab 测试

$ ab -t 20 -c 10 http://192.168.1.104:9510/

三、利用文件排他锁(非阻塞模式)

非阻塞模式下,如果进程在获取文件排他锁时,其它进程正在占用锁的话,此进程会马上判断获取锁失败,并且继续往下执行。

示例代码:

<?php
$http = new swoole_http_server("0.0.0.0", 9511);

$http->set(array(
  'reactor_num' => 2, //reactor thread num
  'worker_num' => 4  //worker process num
));

$http->on('request', function (swoole_http_request $request, swoole_http_response $response) {

  $uniqid = uniqid('uid-', TRUE);
  $redis = new Redis();
  $redis->connect('127.0.0.1', 6379);

  $fp = fopen("lock.txt", "w+");

  // 非阻塞模式, 如果不希望 flock() 在锁定时堵塞,则给 lock 加上 LOCK_NB
  if(flock($fp,LOCK_EX | LOCK_NB))  //锁定当前指针
  {
   // 成功取得锁后,放心处理订单
    $rest_count = intval($redis->get("rest_count"));
    $value = "{$rest_count}-{$uniqid}";
    if($rest_count > 0){
      // do something ...
      $rand = rand(100, 1000000);
      $sum=0;
      for ($i=0;$i<$rand;$i++){ $sum+=$i; }

      $redis->lPush('uniqids', $value);
      $redis->decr('rest_count');
    }

   // 订单处理完成后,再释放锁
    flock($fp,LOCK_UN);
  } else {
   // 如果获取锁失败,马上进入这里执行
    echo "{$uniqid} - 系统繁忙,请稍后再试".PHP_EOL;
  }
  fclose($fp);

});

$http->start();

使用 ab 测试

$ ab -t 20 -c 10 http://192.168.1.104:9511/

最后给出三种处理方式的测试结果比较

redis 事务方式:

......
Concurrency Level:   10
Time taken for tests:  20.005 seconds
Complete requests:   17537
Failed requests:    0
Total transferred:   2578380 bytes
HTML transferred:    0 bytes
Requests per second:  876.62 [#/sec] (mean)
Time per request:    11.407 [ms] (mean)
Time per request:    1.141 [ms] (mean, across all concurrent requests)
Transfer rate:     125.86 [Kbytes/sec] received
......

文件排他锁(阻塞模式):

......
Concurrency Level:   10
Time taken for tests:  20.003 seconds
Complete requests:   8205
Failed requests:    0
Total transferred:   1206282 bytes
HTML transferred:    0 bytes
Requests per second:  410.19 [#/sec] (mean)
Time per request:    24.379 [ms] (mean)
Time per request:    2.438 [ms] (mean, across all concurrent requests)
Transfer rate:     58.89 [Kbytes/sec] received
......

文件排他锁(非阻塞模式):

......
Concurrency Level:   10
Time taken for tests:  20.002 seconds
Complete requests:   8616
Failed requests:    0
Total transferred:   1266846 bytes
HTML transferred:    0 bytes
Requests per second:  430.77 [#/sec] (mean)
Time per request:    23.214 [ms] (mean)
Time per request:    2.321 [ms] (mean, across all concurrent requests)
Transfer rate:     61.85 [Kbytes/sec] received
......

经测试结果对比,redis 事务方式优于文件排他锁方式,而文件排他锁方式中,非阻塞模式优于阻塞模式。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

PHP 相关文章推荐
PHP的类 功能齐全的发送邮件类
Oct 09 PHP
一篇不错的PHP基础学习笔记
Mar 18 PHP
PHP 实现多服务器共享 SESSION 数据
Aug 15 PHP
php数据库密码的找回的步骤
Jan 12 PHP
深入php数据采集的详解
Jun 02 PHP
PHP和C#可共用的可逆加密算法详解
Oct 26 PHP
thinkPHP实现的省市区三级联动功能示例
May 05 PHP
PHP实现的最大正向匹配算法示例
Dec 19 PHP
Laravel框架使用Redis的方法详解
May 30 PHP
laravel 数据迁移与 Eloquent ORM的实现方法
Apr 12 PHP
PHP批斗大会之缺失的异常详解
Jul 09 PHP
Yii框架参数配置文件params用法实例分析
Sep 11 PHP
PHP 实现文件压缩解压操作的方法
Jun 14 #PHP
php反射学习之依赖注入示例
Jun 14 #PHP
php反射学习之不用new方法实例化类操作示例
Jun 14 #PHP
PHP反射学习入门示例
Jun 14 #PHP
PHP如何实现阿里云短信sdk灵活应用在项目中的方法
Jun 14 #PHP
PHP中常用的三种设计模式详解【单例模式、工厂模式、观察者模式】
Jun 14 #PHP
PHP面向对象程序设计子类扩展父类(子类重新载入父类)操作详解
Jun 14 #PHP
You might like
PHP 事件机制(2)
2011/03/23 PHP
PHP过滤★等特殊符号的正则
2014/01/27 PHP
php动态生成缩略图并输出显示的方法
2015/04/20 PHP
PHP中预定义的6种接口介绍
2015/05/12 PHP
php使用crypt()函数进行加密
2017/06/08 PHP
PHP大文件分片上传的实现方法
2018/10/28 PHP
Javascript中的window.event.keyCode使用介绍
2011/04/26 Javascript
在IE浏览器中resize事件执行多次的解决方法
2011/07/12 Javascript
js两行代码按指定格式输出日期时间
2011/10/21 Javascript
详解JavaScript函数绑定
2013/08/18 Javascript
使用javascript实现页面定时跳转总结篇
2013/09/21 Javascript
js中通过split函数分割字符串成数组小例子
2013/09/21 Javascript
js和jquery使按钮失效为不可用状态的方法
2014/01/26 Javascript
Node.js的包详细介绍
2015/01/14 Javascript
AngularJS基础知识笔记之过滤器
2015/05/10 Javascript
JS数组array元素的添加和删除方法代码实例
2015/06/01 Javascript
基于Jquery实现表单验证
2020/07/20 Javascript
javascript运算符——逻辑运算符全面解析
2016/06/27 Javascript
jQuery简单实现title提示效果示例
2016/08/01 Javascript
浅谈bootstrap使用中的一些问题以及解决过程
2016/10/18 Javascript
浅析Node.js:DNS模块的使用
2016/11/23 Javascript
详解利用Angular实现多团队模块化SPA开发框架
2017/11/27 Javascript
vue实现移动端悬浮窗效果
2018/12/01 Javascript
Vue 使用Props属性实现父子组件的动态传值详解
2019/11/13 Javascript
[01:13:08]2018DOTA2亚洲邀请赛4.6 淘汰赛 mineski vs LGD 第二场
2018/04/10 DOTA
[43:41]VP vs RNG 2019国际邀请赛淘汰赛 败者组 BO3 第二场 8.21.mp4
2020/07/19 DOTA
[45:52]完美世界DOTA2联赛PWL S3 Forest vs INK ICE 第二场 12.09
2020/12/12 DOTA
python监控网站运行异常并发送邮件的方法
2015/03/13 Python
Python调用adb命令实现对多台设备同时进行reboot的方法
2018/10/15 Python
对python多线程SSH登录并发脚本详解
2019/02/14 Python
Python爬虫beautifulsoup4常用的解析方法总结
2019/02/25 Python
PyQt5笔记之弹出窗口大全
2019/06/20 Python
如何在django中实现分页功能
2020/04/22 Python
庆祝国庆节标语
2014/10/09 职场文书
图文详解nginx日志切割的实现
2022/01/18 Servers
SpringBoot2零基础到精通之数据与页面响应
2022/03/22 Java/Android