PHP "真异步" TrueAsync SAPI 与 NGINX Unit 集成
PHP "真异步" TrueAsync SAPI 与 NGINX Unit 集成
现在的 Web 开发和过去最大的区别是什么?一句话:没人再愿意等服务器响应了。
七八年前,甚至更早的时候,模块加载、组件打包、脚本解释、数据库查询——这些步骤慢一点,对业务和用户也不会造成太大影响。
现在不一样了。Web 开发的核心已经变成了最大化服务器响应速度。这种转变来自网速的提升和单页应用(SPA)的普及。对后端来说,就是要能处理海量的快速请求,还得把负载分配好。
经典的双池架构(请求 worker + 任务 worker)不是凭空出现的。
一个请求一个进程的模型,根本扛不住大批量的轻量请求。该上并发了——一个进程同时处理多个请求。
并发处理带来了新要求:服务器代码要尽可能贴近业务逻辑。以前不是这样的。以前可以用 CGI 或 FPM 把 Web 服务器和脚本文件分得清清楚楚,很优雅。现在这招不好使了。
所以现在的方案要么就是把组件集成得尽量紧密,要么干脆把 Web 服务器当内部模块嵌进去。NGINX Unit 就是这么干的——它把 JavaScript、Python、Go 这些语言直接嵌到 worker 模块里。PHP 也有模块,但一直以来,PHP 在这种直接集成里没捞到什么好处,因为还是一个 worker 只能处理一个请求。
原文 PHP "真异步" TrueAsync SAPI 与 NGINX Unit 集成
集成特性
架构
这个集成分三层:
C 层(nxt_php_sapi.c, nxt_php_extension.c)
- 在 PHP 里注册 TrueAsync SAPI
- 给每个请求创建协程
- 通过
nxt_unit_run()管理事件循环 - 通过
nxt_unit_response_write_nb()实现非阻塞数据传输
PHP 扩展层(NginxUnit 命名空间)
NginxUnit\Request- 请求对象NginxUnit\Response- 响应对象,支持非阻塞发送NginxUnit\HttpServer::onRequest()- 注册请求处理器
用户代码(entrypoint.php)
- 通过
HttpServer::onRequest()注册处理器 - 使用 Request/Response API
- 完全异步执行
请求流程
HTTP 请求 → NGINX Unit → nxt_php_request_handler()
↓
创建协程 (zend_async_coroutine_create)
↓
nxt_php_request_coroutine_entry()
↓
创建 Request/Response 对象
↓
调用 entrypoint.php 中的回调函数
↓
response->write() → nxt_unit_response_write_nb()
↓
response->end() → nxt_unit_request_done()
非阻塞 I/O
调用 $response->write($data) 时:
- 数据通过
nxt_unit_response_write_nb()发送 - 缓冲区满了,剩余数据进
drain_queue - 缓冲区空出来,触发
shm_ack_handler - 异步写入,不阻塞协程
配置
unit-config.json
{
"applications": {
"my-php-async-app": {
"type": "php",
"async": true, // 启用 TrueAsync 模式
"processes": 2, // 工作器数量
"entrypoint": "/path/to/entrypoint.php",
"working_directory": "/path/to/",
"root": "/path/to/"
}
},
"listeners": {
"127.0.0.1:8080": {
"pass": "applications/my-php-async-app"
}
}
}
重要:"async": true 会激活 TrueAsync SAPI,而不是标准的 PHP SAPI。
加载配置
curl -X PUT --data-binary @unit-config.json \
--unix-socket /tmp/unit/control.unit.sock \
http://localhost/config
entrypoint.php
基本结构:
<?php
use NginxUnit\HttpServer;
use NginxUnit\Request;
use NginxUnit\Response;
set_time_limit(0);
// 注册请求处理器
HttpServer::onRequest(static function (Request $request, Response $response) {
// 拿请求数据
$method = $request->getMethod();
$uri = $request->getUri();
// 设响应头
$response->setHeader('Content-Type', 'application/json');
$response->setStatus(200);
// 发数据(非阻塞)
$response->write(json_encode([
'message' => 'Hello from TrueAsync!',
'method' => $method,
'uri' => $uri
]));
// 结束响应
$response->end();
});
API 参考
Request
getMethod(): string- HTTP 方法(GET、POST 等)getUri(): string- 请求 URIgetRequestContext(): ?mixed- 请求上下文(TODO)getRequestContextParameters(): ?mixed- 上下文参数(TODO)createResponse(): Response- 创建 Response 对象(通常不需要)
Response
setStatus(int $code): bool- 设置 HTTP 状态码setHeader(string $name, string $value): bool- 添加响应头write(string $data): bool- 发送数据(非阻塞操作)end(): bool- 完成响应并释放资源
注意:
setStatus()和setHeader()要在第一次write()之前调用- 调用过
write()后,响应头就发出去了 end()必须调用,完成请求
生命周期
HttpServer::onRequest(function (Request $req, Response $resp) {
// 1. 响应头还能改
$resp->setStatus(200);
$resp->setHeader('Content-Type', 'text/plain');
// 2. 第一次 write() 把响应头发出去了
$resp->write('Hello ');
// 3. 现在响应头改不了了
// $resp->setHeader() → 报错!
// 4. 可以继续写数据
$resp->write('World!');
// 5. 结束请求(必须调!)
$resp->end();
});
运行和测试
启动 NGINX Unit
./build/sbin/unitd \
--no-daemon \
--log /tmp/unit/unit.log \
--state /tmp/unit \
--control unix:/tmp/unit/control.unit.sock \
--pid /tmp/unit/unit.pid \
--modules ./build/lib/unit/modules
重要:--modules 参数必须加,用来加载 PHP 模块。
查看日志
tail -f /tmp/unit/unit.log
测试
curl http://127.0.0.1:8080/
响应:
{
"message": "Hello from NginxUnit TrueAsync HttpServer!",
"method": "GET",
"uri": "/",
"timestamp": "2025-10-04 15:30:00"
}
负载测试
wrk -t4 -c100 -d30s http://127.0.0.1:8080/
调试
GDB
gdb ./build/sbin/unitd
(gdb) set follow-fork-mode child
(gdb) run --no-daemon --log /tmp/unit/unit.log ...
设置断点
break nxt_php_request_handler
break nxt_php_request_coroutine_entry
break nxt_unit_response_write_nb
实用命令
# 停止所有 NGINX Unit 进程
pkill -9 unitd
# 检查控制套接字
ls -la /tmp/unit/control.unit.sock
# 获取当前配置
curl --unix-socket /tmp/unit/control.unit.sock http://localhost/config
内部实现
初始化
nxt_php_extension_init()在NginxUnit命名空间注册类- worker 启动时加载
entrypoint.php HttpServer::onRequest()把回调存到nxt_php_request_callback
请求处理
- NGINX Unit 调用
nxt_php_request_handler(req) - 创建协程:
zend_async_coroutine_create(nxt_php_request_coroutine_entry) - 协程指针存到
req - 协程加入激活队列
- 控制权回到事件循环
nxt_unit_run()
协程激活
- 事件循环调用
nxt_unit_response_buf_alloc回调 - 回调通过
zend_async_coroutine_activate()激活协程 - 执行
nxt_php_request_coroutine_entry() - 创建 PHP Request/Response 对象
- 调用用户回调
response->end()后协程结束
异步发送
response->write()→nxt_unit_response_write_nb()- 没发完,剩下的进
drain_queue - 缓冲区空了,触发
shm_ack_handler() shm_ack_handler继续写,需要的话调end()
未来计划
- 实现
Request::getRequestContext() - 添加请求头支持
- 添加 POST 数据解析
- WebSocket 支持
- 流式响应
总结
NGINX Unit TrueAsync PHP 集成让 PHP 真正拥有了异步处理能力。通过协程机制,单个进程可以同时处理多个请求,这在过去是无法想象的。
对于 PHP 生态来说,这是一个重要的转折点。传统的 PHP-FPM 模式下,每个请求独占一个进程,在高并发场景下资源消耗巨大。现在有了 TrueAsync,PHP 可以像 Node.js、Go 那样高效处理并发请求,同时保持语言本身的简洁性。
虽然目前还有一些功能在开发中,比如完整的请求头支持、POST 数据解析、WebSocket 等,但现有的功能已经足够构建高性能的 API 服务。非阻塞 I/O、协程调度、事件循环——这些核心机制都已经就位。
对于需要处理高并发请求的 PHP 应用,特别是 API 服务、微服务架构,NGINX Unit TrueAsync 提供了一个值得认真考虑的选择。它不需要改变太多现有代码结构,却能带来性能上的显著提升。
PHP "真异步" TrueAsync SAPI 与 NGINX Unit 集成的更多相关文章
- Nginx unit 源码安装初体验
Nginx unit 源码安装初体验 上次介绍了从yum的安装方法(https://www.cnblogs.com/wang-li/p/9684040.html),这次将介绍源码安装,目前最新版为1. ...
- nginx Unit 服务器
转自: https://github.com/nginx/unit/pull/18/ 感谢: https://www.v2ex.com/t/389528 English 简体中文 繁體中文 NGINX ...
- PHP fsockopen 异步调用接口在nginx上偶尔实效的情况
private function fsock_asy_do($get){ $fp = fsockopen("ssl://www.xxx.com", 443, $errno, $er ...
- svn + nginx unit + python3自动化发布web服务方法
本周将python web服务管理更换成nginx unit以后发现接口性能有了明显的提升,访问速度快了不少.不过有个很大的问题就是使用svn自动化发布以后,服务并没有刷新使用新的代码运行,而又不懂得 ...
- 关于nginx unit服务非正常关闭后,无法重新启动问题的处理
昨天在前领导技术大牛吕哥的帮忙下,python服务管理从nginx+supervisor+uwsgi+python3改为了轻便结构nginx + unit + python3,部署和配置起来顿时轻松起 ...
- nginx unit PHP
2018-12-26 14:20:33 星期三 综述: nginx unit php 的关系: nginx -> 转发请求到 8300端口 -> unit 转发 8300 收到的请求 -& ...
- nginx unit nodejs 模块试用(续)
最新(应该是18 年了)nginx unit 发布了新的版本,对于nodejs 的支持有很大的改进,上次测试过,问题还是 比较多,这次使用新版本在测试下对于nodejs 的支持,以及以前block ...
- nginx unit nodejs 模块试用
unit 对于nodejs 的支持是在10.25 发布的,基本能用,但是依然有好多问题,当前在测试的时候就发现,请求之后会block , 相关的issue 已经有人反馈了,最好使用源码编译,方便测 ...
- nginx unit java 试用
unit 当前已经支持java了,当时支持基于servlet 的开发模式,以下是一个简单的学习 基于官方的demo 环境准备 docker-compose文件 version: "3&q ...
- nginx lua集成kafka
NGINX lua集成kafka 第一步:进入opresty目录 [root@node03 openresty]# cd /export/servers/openresty/ [root@node03 ...
随机推荐
- Failed to start LVS and VRRP High Availability Monitor.-keepalived--九五小庞
Keepalive启动报错,Fail to start LVS and VRRP High Availability Monitor. 输入:systemctl status keepalived,显 ...
- Docker问题解决:Error response from daemon: Get https://registry-1.docker.io/v2/: dial tcp: lookup registry-1.docker.io: no such host
vi /etc/resolv.conf 添加 nameserver 8.8.8.8
- selenium,web自动化测试
如何delphi7中添加TClientSocket 组件 在Component->instal package选择ADD添加borland/delphi/bin/dclsockets70.bpl ...
- ansible-playbook基础
主机与用户 你可以为 playbook 中的每一个 play,个别地选择操作的目标机器是哪些,以哪个用户身份去完成要执行的步骤(called tasks). hosts 行的内容是一个或多个组或主机的 ...
- ts 常用配置
这里记录一些常用的 ts config. strictNullChecks "strictNullChecks": true 严格区分 undefined 和 null let w ...
- 1003 Express Mail Taking
http://acm.hdu.edu.cn/contests/contest_showproblem.php?pid=1003&cid=909 Express Mail Taking Time ...
- flask四件套、flask注册路由、flask获取post请求传入的数据、flask开启debug模式
flask四件套 返回模板:return render_template('detail.html', **{'user': user}) # context跟djagno不一样 返回重定向:retu ...
- 记录一下Windows系统下的命令行参数的字符个数限制
记得在三年前,我们在做应用升级,选型了通过启动传参的方式,客户端通过传递参数给更新器 参数,执行应用升级.一开始更新的应用不多,参数的字符个数也是在1000以内,没有发现问题.后来随着全家桶应用的新增 ...
- [题解]P4597 序列 sequence
P4597 序列 sequence 是CF13C Sequence的加强版,\(N\leq 5*10^5\). 如果想了解\(O(N^2)\)的DP解法请看此文. 给定\(N\)个数,每次操作可以选其 ...
- 告别复杂配置!使用 1Panel 运行环境功能轻松搭建 Java 应用
一.引言:Java 部署之痛与新生 作为开发者,你是否经历过这些场景? 为不同Java版本频繁切换环境变量 在服务器手动安装Nginx或Tomcat配置端口 调试环境差异导致的"在我本地是好 ...