千家信息网

PHP并发场景的解决方案是什么

发表于:2025-11-07 作者:千家信息网编辑
千家信息网最后更新 2025年11月07日,本篇文章给大家分享的是有关PHP并发场景的解决方案是什么,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。在秒杀,抢购等并发场景下,可能会出
千家信息网最后更新 2025年11月07日PHP并发场景的解决方案是什么

本篇文章给大家分享的是有关PHP并发场景的解决方案是什么,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

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

列出常见的解决方案有:

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

  • 一、利用 Redis 事务特征

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

    示例代码:

    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/

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

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

示例代码:

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/

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

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

示例代码:

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: 10Time taken for tests: 20.005 secondsComplete requests: 17537Failed requests: 0Total transferred: 2578380 bytesHTML transferred: 0 bytesRequests 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: 10Time taken for tests: 20.003 secondsComplete requests: 8205Failed requests: 0Total transferred: 1206282 bytesHTML transferred: 0 bytesRequests 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: 10Time taken for tests: 20.002 secondsComplete requests: 8616Failed requests: 0Total transferred: 1266846 bytesHTML transferred: 0 bytesRequests 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并发场景的解决方案是什么,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注行业资讯频道。

进程 处理 文件 模式 阻塞 订单 事务 方式 测试 方案 解决方案 场景 成功 代码 数据 用户 示例 队列 原子 指针 数据库的安全要保护哪些东西 数据库安全各自的含义是什么 生产安全数据库录入 数据库的安全性及管理 数据库安全策略包含哪些 海淀数据库安全审计系统 建立农村房屋安全信息数据库 易用的数据库客户端支持安全管理 连接数据库失败ssl安全错误 数据库的锁怎样保障安全 服务器监控有两个时间设置 众搜网络技术有限公司 泰兴多功能网络技术多少钱 大同机械网络安全质量服务 数据表格如何保存到数据库 虹口区特定软件开发服务结构设计 上海新时代网络技术有限公司 上海的网络安全比赛 计算机网络技术实践报告结论 河南省网络安全办温学生处长 刺激战场海岛服务器在哪里 数据库检查值是否存在 国际上最著名的三大核苷酸数据库 网站服务器租用还是买 诚实守信促进网络安全手抄报内容 福州有哪些软件开发公司吗 衡水网络安全新闻 软件开发工程师的工作 数据库原理试题及答案解析 网络技术和数据库哪个容易考 龙芯服务器中国 数据库删除一条消息怎么恢复 软件开发员是干嘛的 租用服务器需要特许经营吗 数据库索引的四种类型 常州网络安全攻防实战演练 维护网络安全的代名词 北京展厅互动软件开发 网络安全联盟联系方式 大学信息与网络技术基础书
0