PHP无原生异步运行时,async/await非语言特性而是协程封装;可行方案为Swoole、ReactPHP或消息队列+Worker,需依任务类型选择并注意协程API替换、序列化限制及内存泄漏等关键细节。

PHP 本身没有原生异步运行时,别被“async/await”误导
PHP 的 async 和 await(如 amphp、Swoole 4.8+ 的协程语法糖)不是语言级特性,而是底层事件循环 + 协程调度的封装。直接写 async function 在 CLI 或 FPM 下会报错:Fatal error: Uncaught Error: Call to undefined function async()。真要异步,必须换运行环境或加扩展。
三种可行路径:Swoole、ReactPHP、消息队列 + Worker
选哪条取决于你已有架构和任务类型:
-
短时高频 I/O 密集型(如并发调多个 HTTP 接口、查 Redis)→ 用
Swoole\Coroutine,启动快、协程开销小,但要求 PHP 运行在 Swoole Server 或swoole-cli -
需要兼容传统 PHP 生态(比如不能改 Web 服务器)→ 用
ReactPHP,基于回调和 Promise,不依赖扩展,但需重写阻塞逻辑,学习成本高 -
长耗时、可延迟、需失败重试(如发邮件、生成报表、视频转码)→ 不该在请求中做,应投递到
Redis或RabbitMQ,由独立php artisan queue:work(Laravel)或自研worker.php消费
用 Swoole 启动一个真正异步 HTTP 请求
注意:Swoole\Http\Client 已废弃,必须用 Swoole\Coroutine\Http\Client,且只能在协程上下文里调用:
set(['timeout' => 5]);
$client->get('/delay/2');
echo "status: {$client->statusCode}\n"; // 不会阻塞后续协程
});
// 必须显式启动调度器,否则协程不执行
Swoole\Event::wait();
常见坑:file_get_contents、curl_exec、PDO 查询等默认仍是同步阻塞,即使在 Coroutine::create 里也会让整个协程挂起——必须换用 Swoole 提供的协程版 API。
立即学习“PHP免费学习笔记(深入)”;
消息队列方案最容易落地,但也最容易漏掉关键点
以 Laravel + Redis 为例,dispatch(new SendEmailJob()) 看似异步,实际只是把任务序列化后推入 Redis。真正执行靠 queue:work 进程。容易忽略的细节:
- Web 请求进程和 Worker 进程是分离的,
$_SESSION、static变量、未关闭的 PDO 连接不会自动传递 - Job 类必须可序列化,闭包、资源句柄(如
cURL handle)、$this引用不可带入 - Redis 队列没持久化,服务重启可能丢任务;要用
redis.conf开启appendonly yes,或直接切database驱动存 MySQL - Worker 进程常驻内存,记得在
__destruct或finally里释放大对象、关闭文件句柄,否则内存缓慢泄漏
异步不是加个 dispatch 就完事,得盯住数据边界、生命周期和错误兜底——尤其是那个没人看的日志文件。











