当前位置: 首页 > news >正文

PHP是如何并行异步处理HTTP请求的?

文章精选推荐

1 JetBrains Ai assistant 编程工具让你的工作效率翻倍
2 Extra Icons:JetBrains IDE的图标增强神器
3 IDEA插件推荐-SequenceDiagram,自动生成时序图
4 BashSupport Pro 这个ides插件主要是用来干嘛的 ?
5 IDEA必装的插件:Spring Boot Helper的使用与功能特点
6 Ai assistant ,又是一个写代码神器
7 Cursor 设备ID修改器,你的Cursor又可以继续试用了

文章正文

在 PHP 中,由于其传统的同步阻塞模型,实现并行异步处理 HTTP 请求并不像其他语言(如 Go 或 Node.js)那样直接。不过,仍然可以通过一些扩展和工具来实现并行异步处理。以下是几种常见的方法:

1. 使用 cURL 的多线程功能

PHP 的 cURL 扩展支持多线程处理,可以通过 curl_multi_* 系列函数实现并行 HTTP 请求。

示例代码:
$urls = [
    'https://example.com/api/1',
    'https://example.com/api/2',
    'https://example.com/api/3',
];

$mh = curl_multi_init(); // 初始化多线程 cURL
$handles = [];

foreach ($urls as $url) {
    $ch = curl_init();
    curl_setopt($ch, CURLOPT_URL, $url);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
    curl_multi_add_handle($mh, $ch); // 将单个 cURL 句柄添加到多线程中
    $handles[] = $ch;
}

$running = null;
do {
    curl_multi_exec($mh, $running); // 执行并行请求
    curl_multi_select($mh); // 等待活动
} while ($running > 0);

$responses = [];
foreach ($handles as $ch) {
    $responses[] = curl_multi_getcontent($ch); // 获取每个请求的响应
    curl_multi_remove_handle($mh, $ch); // 移除句柄
    curl_close($ch);
}

curl_multi_close($mh); // 关闭多线程 cURL

print_r($responses);
优点:
  • 原生支持,无需额外扩展。
  • 可以并行处理多个 HTTP 请求。
缺点:
  • 代码复杂度较高。
  • 需要手动管理句柄和状态。

2. 使用 Guzzle 异步客户端

Guzzle 是一个流行的 PHP HTTP 客户端库,支持异步请求。

示例代码:
require 'vendor/autoload.php';

use GuzzleHttp\Client;
use GuzzleHttp\Promise;

$client = new Client();

$urls = [
    'https://example.com/api/1',
    'https://example.com/api/2',
    'https://example.com/api/3',
];

$promises = [];
foreach ($urls as $url) {
    $promises[] = $client->getAsync($url); // 发起异步请求
}

$responses = Promise\Utils::settle($promises)->wait(); // 等待所有请求完成

foreach ($responses as $response) {
    if ($response['state'] === 'fulfilled') {
        echo $response['value']->getBody() . "\n"; // 输出响应内容
    } else {
        echo 'Request failed: ' . $response['reason']->getMessage() . "\n";
    }
}
优点:
  • 代码简洁,易于使用。
  • 支持并发请求和异步处理。
缺点:
  • 需要安装 Guzzle 库。

3. 使用 Swoole 扩展

Swoole 是一个高性能的 PHP 扩展,支持异步、协程和并行处理。

示例代码:
Swoole\Runtime::enableCoroutine(); // 启用协程

$urls = [
    'https://example.com/api/1',
    'https://example.com/api/2',
    'https://example.com/api/3',
];

$responses = [];

go(function () use ($urls, &$responses) {
    $client = new Swoole\Coroutine\Http\Client('example.com', 443, true);
    foreach ($urls as $url) {
        $client->get($url);
        $responses[] = $client->body;
    }
});

Swoole\Event::wait(); // 等待所有协程完成

print_r($responses);
优点:
  • 高性能,支持协程和异步 I/O。
  • 适合高并发场景。
缺点:
  • 需要安装 Swoole 扩展。
  • 学习曲线较高。

4. 使用 ReactPHP

ReactPHP 是一个基于事件驱动的 PHP 库,支持异步编程。

示例代码:
require 'vendor/autoload.php';

use React\EventLoop\Factory;
use React\HttpClient\Client;
use React\HttpClient\Response;

$loop = Factory::create();
$client = new Client($loop);

$urls = [
    'https://example.com/api/1',
    'https://example.com/api/2',
    'https://example.com/api/3',
];

foreach ($urls as $url) {
    $request = $client->request('GET', $url);
    $request->on('response', function (Response $response) {
        $response->on('data', function ($chunk) {
            echo $chunk;
        });
    });
    $request->end();
}

$loop->run();
优点:
  • 基于事件驱动,适合异步编程。
  • 支持长连接和流式处理。
缺点:
  • 需要安装 ReactPHP 库。
  • 代码复杂度较高。

5. 使用多进程(pcntl 扩展)

PHP 的 pcntl 扩展支持多进程编程,可以通过创建子进程来实现并行处理。

示例代码:
$urls = [
    'https://example.com/api/1',
    'https://example.com/api/2',
    'https://example.com/api/3',
];

$children = [];

foreach ($urls as $url) {
    $pid = pcntl_fork();
    if ($pid == -1) {
        die('Could not fork');
    } elseif ($pid) {
        $children[] = $pid; // 父进程记录子进程 ID
    } else {
        // 子进程处理请求
        echo file_get_contents($url) . "\n";
        exit(); // 子进程退出
    }
}

// 父进程等待所有子进程完成
foreach ($children as $pid) {
    pcntl_waitpid($pid, $status);
}
优点:
  • 真正的并行处理。
  • 适合 CPU 密集型任务。
缺点:
  • 需要 pcntl 扩展。
  • 进程间通信复杂。

总结

  • cURL 多线程:适合简单的并行 HTTP 请求。
  • Guzzle:代码简洁,适合大多数场景。
  • Swoole:高性能,适合高并发场景。
  • ReactPHP:基于事件驱动,适合异步编程。
  • 多进程:适合 CPU 密集型任务,但复杂度较高。

根据具体需求选择合适的方法即可。

相关文章:

  • CSS基础入门——盒模型与布局
  • CES Asia 2025“传播势能放大器”:科技与环保的双重盛宴
  • 【gopher的java学习笔记】什么是classLoader
  • 网络工程师 (43)IP数据报
  • FormCreate设计器v5.6发布—AI智能表单助理正式上线!
  • Linux 外设驱动 应用 6 陀螺仪实验
  • 【算法】链表
  • Unity Shader学习6:多盏平行光+点光源 ( 逐像素 ) 前向渲染 (Built-In)
  • Word中接入大模型教程
  • 深度学习框架探秘|Keras:深度学习的魔法钥匙
  • 最新智能优化算法: 贪婪个体优化算法(Greedy Man Optimization Algorithm,GMOA)求解23个经典函数测试集,MATLAB代码
  • Vivado生成edif网表及其使用
  • 高效学习方法分享
  • Python学习心得常见的异常
  • Redis 主从复制
  • Mybatis后端数据库查询多对多查询解决方案
  • Lombok注解@EqualsAndHashCode
  • apache artemis安装
  • Git 使用指南:避免使用 merge 的完整流程
  • python学opencv|读取图像(六十七)使用cv2.convexHull()函数实现图像轮廓凸包标注
  • 茅台总经理到访五粮液:面对白酒行业周期性调整,需要团结一心的合力
  • 新疆多地市民拍到不明飞行物:几秒内加速消失,气象部门回应
  • 戛纳打破“疑罪从无”惯例,一法国男演员被拒之门外
  • 韶关一企业将消防安装工程肢解发包,广东住建厅:罚款逾五万
  • 再现五千多年前“古国时代”:凌家滩遗址博物馆今开馆
  • 梅花奖在上海|舞剧《朱鹮》,剧里剧外都是生命的赞歌