常见解决方法如下:

1、使用队列,额外起一个进程处理队列,并发请求都放到队列中,由额外进程串行处理,并发问题就不存在了,但是要额外进程支持以及处理延迟严重,本文不先不讨论这种方法。

2、利用数据库事务特征,做原子更新,此方法需要依赖数据库的事务特性。

3、借助文件排他锁,在处理下单请求的时候,用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教程

更多相关文章

  1. 用PHP+Redis实现延迟任务 实现自动取消订单(详细教程)
  2. PHP 多进程和多线程的优缺点
  3. 解决CentOS7中php-fpm进程数过多导致服务器内存资源消耗较大的问
  4. 订单模块数据库表解析(三)
  5. 订单模块数据库表解析(二)
  6. 订单模块数据库表解析(一)
  7. 商品管理 数据库表结构 图片 功能结构 图片 订单管理 数据
  8. Linux进程管理
  9. 多线程学习(一) 线程与进程的理解

随机推荐

  1. Android 让一个控件或布局居于底部的办法
  2. Android蓝牙开发浅析
  3. android EditText输入限制
  4. Android中的消息机制-源码分析
  5. Android系统自带样式(android:theme)
  6. Android本地动作
  7. Android(安卓)禁止屏幕旋转和重启Activit
  8. Android(安卓)游戏开发中横竖屏切换问题
  9. Android提供的系统服务之--WindowManager
  10. [转]: 两分钟彻底让你明白Android(安卓)A