- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章PHP 并发场景的几种解决方案由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
在秒杀,抢购等并发场景下,可能会出现超卖的现象,在PHP语言中并没有原生提供并发的解决方案,因此就需要借助其他方式来实现并发控制.
列出常见的解决方案有:
1、利用 Redis 事务特征 。
redis 事务是原子操作,可以保证订单处理的过程中数据没有被其它并发的进程修改.
示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
<?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 测试 。
1
|
$ ab -t 20 -c 10 http:
//192.168.1.104:9509/
|
2、利用文件排他锁(阻塞模式) 。
阻塞模式下,如果进程在获取文件排他锁时,其它进程正在占用锁的话,此进程会挂起等待其它进程释放锁后,并自己获取到锁后,再往下执行.
示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
<?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 测试 。
1
|
$ ab -t 20 -c 10 http:
//192.168.1.104:9510/
|
3、利用文件排他锁(非阻塞模式) 。
非阻塞模式下,如果进程在获取文件排他锁时,其它进程正在占用锁的话,此进程会马上判断获取锁失败,并且继续往下执行.
示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
|
<?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 测试 。
1
|
$ ab -t 20 -c 10 http:
//192.168.1.104:9511/
|
最后给出三种处理方式的测试结果比较 。
redis 事务方式:
1
2
3
4
5
6
7
8
9
10
11
12
|
......
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
......
|
文件排他锁(阻塞模式):
1
2
3
4
5
6
7
8
9
10
11
12
|
......
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
......
|
文件排他锁(非阻塞模式):
1
2
3
4
5
6
7
8
9
10
11
12
|
......
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 事务方式优于文件排他锁方式,而文件排他锁方式中,非阻塞模式优于阻塞模式.
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我.
原文链接:https://segmentfault.com/a/1190000019455439 。
最后此篇关于PHP 并发场景的几种解决方案的文章就讲到这里了,如果你想了解更多关于PHP 并发场景的几种解决方案的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
websocket的用途/场景 先总结:高即时性服务,比如聊天室的群聊,server顺序收到了张三,李四的消息,立即就推送给王五,不能让王五等半天。 Ajax也可以一秒一刷,让王五去问张三说话没,如果
前端的工作过程里,本地开发、提供测试环境,总得有个用着顺手的服务器软件,这个场景里nginx很流行。 介绍两个好用的配置项:rewrite try_files @xxxx rewrite 比较
我有一个场景的两个不同角度的 2 个视频文件,我想重建场景的 3D 估计。它类似于 3D 传感器的作用(例如 Kinect、PrimeSense)。我正在寻找一个库,甚至是一个完善的机器视觉算法,以便
我已阅读RebaseProject页面并尝试了一个不平凡的例子(不是对一个完整的分支进行 rebase )。这与 rebase D 的情况类似我场景B。 这是rebase之前的情况: default
有没有办法将我的场景保存在 JavaFx 应用程序中单独的 Java 文件中?我尝试过这样的事情: public class MyApp extends Application { pri
我有这样的场景:用户想要查看大量有关自己的信息。例如:年龄、姓名、地位、收入、工作、爱好、 child 的名字、妻子的名字、酋长的名字、祖父/祖母的名字。大约 50 个变量。他可以选择任何变量来显示信
我希望有人能帮助我解决这个问题:我有一个包含条目的表。我想执行查询并根据模式获取得分最高的记录。模式将是:如果我的话按原样出现,那么该条目的分数将是最高的。如果该单词出现在句子中,则该条目的分数将低于
我正在尝试在我的应用程序委托(delegate)方法中实现一些逻辑。了解当前正在运行哪种场景将非常有帮助。 [[CCDirector sharedDirector] runningScene] 返回当
好的,这是一个有趣的。我有 2 个表:tbl_notes、tbl_notes_categories 简单地说,tbl_notes 有一个 categoryid,我将 2 个表与该 ID 相关联。所以,
我有一个使用并行运行的 Specflow、selenium、NUnit 的测试解决方案在 AssemblyInfo 中添加了这个:[程序集:Parallelizable(ParallelScope.F
我正在尝试弄清楚如何在 SpriteKit 中添加更多场景。如果我在 GameViewController 中使用 SpriteKit 生成的行 if let scene = GameScene.un
目录 1、业务背景 2、场景分析 3、流程设计 1、业务流程 2、导入流程
我是 Unity 的新手,所以修复起来可能非常简单。我使用了一个 3D Google SketchUp 模型,我想让玩家环顾模型。 super 简单。 我添加了 3D 平面,添加了相机并更新了设置以支
我需要标记要跳过的某些测试。但是,有些测试是参数化的,我只需要能够跳过某些场景。 我使用 py.test -m "hermes_only" 调用测试或 py.test -m "not hermes_o
我已经开始使用 SpecFlow 并想知道是否可以在规范之间重用场景 基本上我的想法是这样的(我可能从根本上是错误的:)) 我编写了一项功能来验证导航。 功能:导航 I should be able
在编写验证输入表单上的信息的 BDD 场景时,您将如何列出规则。 选项是: 1) 每个规则一个场景 2)场景大纲,每个领域和规则的例子 我们如何说某些不在特定字符集中的无效内容,例如: 鉴于我输入了一
我们如何使用 StoryQ 来测试预期出现异常的场景? 最佳答案 就实际代码而言,在测试代码的 .Then 部分,您需要创建一个 Action 或 Func 来确定正在测试的内容,然后在代码的 .Th
完成快速初学者努力通过点击按钮向场景添加节点。 我知道我可以使用点击手势来获取点击坐标并执行点击测试,然后在点击的 3D 空间中放置一个对象。但是,我想在设备屏幕的中央显示一个球体或十字准线,当点击屏
如何在表格中传递空格? Background: Given the following books |Author |(here several spaces)
我正在尝试从 Eric Haines' Standard Procedural Database (SPD) 渲染“mount”场景,但折射部分就是不想配合。我已经尝试了所有我能想到的方法来修复它。
我是一名优秀的程序员,十分优秀!