PHP开发中解决并发问题的几种实现方法分析

发布网友 发布时间:2022-04-21 18:20

我来回答

2个回答

懂视网 时间:2022-04-06 05:11

php实现并发的方法:首先写好针对第三方的请求接口;然后使用并发请求这些第三方的接口,并发代码如“function request(){...}”;最后使用php实现循环请求接口即可。

推荐:《PHP视频教程》

php实现并发

使用场景

如果你第三方的多个接口,需要循环请求,这时候就需要并发处理,因为 php 是单进程,顺序执行的程序,如果有一个接口挂了,那么后面的接口也不能得到请求,或者某个接口等待时间过长,下面的接口也需要等待时间,那么十多个接口每个接口等待1S,十个接口就需要10S才能完成一次请求。这时候,就需要使用并发请求

使用方案

可以先写好针对某个第三方的请求接口,然后使用并发请求这些第三方的接口

并发代码

function request(){
$srart_time = microtime(TRUE);
$webSiteUrl = 'http://test/';
/**
 * 结算
 */
$alls = [
 $webSiteUrl . "1.php",
 $webSiteUrl . "2.php",
];
//1 创建批处理cURL句柄
$chHandle = curl_multi_init();
$chArr = [];
//2.创建多个cURL资源
foreach ($alls as $Url) {
 $chArr[$Url] = curl_init();
 curl_setopt($chArr[$Url], CURLOPT_URL, $Url);
 curl_setopt($chArr[$Url], CURLOPT_RETURNTRANSFER, 1);
 curl_setopt($chArr[$Url], CURLOPT_TIMEOUT, 1);
 curl_multi_add_handle($chHandle, $chArr[$Url]); //2 增加句柄
}
$active = null;
/**常量
 * CURLM_CALL_MULTI_PERFORM==-1
 * // CURLM_OK == 0
 **/
do {
 $mrc = curl_multi_exec($chHandle, $active); //3 执行批处理句柄
} while ($mrc == CURLM_CALL_MULTI_PERFORM); //4
//4 $active 为true,即$chHandle批处理之中还有$ch句柄正待处理,$mrc==CURLM_OK,即上一次$ch句柄的读取或写入已经执行完毕。
while ($active && $mrc == CURLM_OK) {
 if (curl_multi_select($chHandle) != CURLM_CALL_MULTI_PERFORM) {//$chHandle批处理中还有可执行的$ch句柄,curl_multi_select($chHandle) != -1程序退出阻塞状态。
 do {
  $mrc = curl_multi_exec($chHandle, $active);//继续执行需要处理的$ch句柄。
 } while ($mrc == CURLM_CALL_MULTI_PERFORM);
 }
}
foreach ($chArr as $k => $ch) {
// $result[$k] = curl_multi_getcontent($ch); //5 获取句柄的返回值,不需要
 curl_multi_remove_handle($chHandle, $ch);//6 将$chHandle中的句柄移除
}
curl_multi_close($chHandle); //7 关闭全部句柄
$end_time = microtime(TRUE);
echo sprintf("use time:%.3f s", $end_time - $srart_time);
}

php 实现循环请求接口

这种情况针对的是循环需要在1S内执行的程序,如果执行频率超过1S,则直接使用linux 的计划任务即可

// 无时间执行此程序
set_time_limit(0);
// 设置默认时区 北京时间
date_default_timezone_set('PRC');
do {
 // 如果不存在 stop 文件,则程序停止 并且记录停止时间
 if (!file_exists(dirname(__FILE__) . '/stop')) {
 $handle = fopen('./ceshi.log', 'a+');
 fwrite($handle, '程序停止时间: ' . date('Y-m-d H:i:s') . "
");
 fclose($handle);
 exit();
 }
 sleep(4);
 try {
 requestNewOpen();
 } catch (Exception $exception) {
 $handle = fopen('./ceshi.log', 'a+');
 fwrite($handle, 'catch : ' . $exception->getMessage() . "
");
 fclose($handle);
 }
} while (true);

热心网友 时间:2022-04-06 02:19

方案一:使用文件锁排它锁
flock函数用于获取文件的锁,这个锁同时只能被一个线程获取到,其它没有获取到锁的线程要么阻塞,要么获取失败
在获取到锁的时候,先查询库存,如果库存大于0,则进行下订单操作,减库存,然后释放锁
方案二:使用队列
将用户的下单请求依次存入一个队列中,后台用一个单独的进程处理队列中的下单请求

声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com