workerman 不支持协程,它基于多进程和回调式事件循环实现异步;swoole 协程由 c 层调度,支持同步风格写异步逻辑;二者设计目标不同,混用易引发隐性依赖和运行时错误。

Workerman 根本没有协程
这是最常被误解的一点:Workerman 不提供协程能力,它用的是多进程 + 回调式事件循环(基于 select/epoll),所有异步操作都靠注册回调函数实现。你写 onMessage、onClose,本质是监听 socket 事件后触发 PHP 函数——没有协程栈、没有 go()、不能 yield、也不支持同步风格写异步逻辑。
常见错误现象:
– 试图在 Workerman 中使用 Swoole\Coroutine\run() 或 go(),直接报错 Class 'Swoole\Coroutine' not found
– 误以为 AsyncTcpConnection 是“协程连接”,其实只是基于 stream_select 的非阻塞连接 + 回调封装
- Workerman 的“异步 MySQL/Redis”组件,底层仍是阻塞 socket + 手动轮询 + 回调调度,不是真正的协程 IO
- 它依赖
pcntl和posix扩展做进程管理,但和协程无关 - 如果你看到某篇教程说“Workerman 协程化开发”,基本是混淆了概念,或偷偷集成了 Swoole 扩展
Swoole 协程是原生 C 层调度的轻量级执行单元
Swoole 的协程(Swoole\Coroutine)由 C 扩展在内核层实现,每个协程共享一个 PHP 进程,但拥有独立的栈空间(默认仅 8KB)、独立的上下文和 fiber 调度器。它能让传统同步代码(比如 mysqli_query、file_get_contents)在协程环境下自动变成非阻塞调用。
典型使用场景:
– 在 HTTP 请求处理中串行调用 Redis + MySQL + 第三方 API,不用写回调嵌套
– 使用 Swoole\Coroutine\MySQL 或 Swoole\Coroutine\Redis,接口几乎和同步客户端一致
- 协程切换开销
,远低于进程/fork,也比 PHP 用户态协程(如 <code>amphp)更底层、更稳定 - 必须在
Swoole\Coroutine\run()或go()启动的上下文中才能生效;在普通 CLI 或 FPM 环境下无效 - 注意:协程内不能调用某些阻塞函数(如
sleep()、usleep()),应改用Swoole\Coroutine::sleep()
为什么 Workerman 不加协程?这不是“落后”,而是设计取舍
Workerman 的目标是“让 PHP 开发者不碰底层也能跑起长连接服务”,它的纯 PHP 实现带来的是可读性、可调试性和部署自由度——你可以用 Xdebug 断点跟踪 onMessage,可以单步进到协议解析逻辑,也可以直接修改源码加日志。而 Swoole 协程依赖扩展,调试需配合 xdebug.mode=develop,coverage 且无法进入 C 层调度逻辑。
性能与兼容性影响:
– Workerman 单连接内存占用略高(48MB/万连接 vs Swoole 32MB/万连接),但对中小项目差异不敏感
– 它天然兼容 HHVM、旧版 PHP(如 7.2),而 Swoole 5.x 已要求 PHP 8.0+
- 如果你的项目只需 WebSocket 聊天室、设备心跳上报、简单信令转发,Workerman 的回调模型足够清晰、够稳、够快
- 一旦涉及密集 IO 编排(如网关聚合多个微服务响应)、复杂状态机(如订单支付流程)、或需复用大量同步生态库(Laravel Scout、Monolog),Swoole 协程才是更自然的选择
混用 Workerman 和 Swoole 的坑:别让 ext-swoole 变成隐藏依赖
有些团队为“兼顾易用和性能”,在 Workerman 项目里悄悄装上 swoole 扩展,然后在部分 Worker 中用 go() 调用协程客户端——这看似双赢,实则埋雷。
常见错误现象:
– 本地开发没装 Swoole,CI 测试通过,上线后 Class 'Swoole\Coroutine' not found 直接崩
– Workerman 主进程 fork 出的子进程,若未显式调用 Swoole\Coroutine\run(),协程上下文丢失,go() 启动失败或静默降级为同步阻塞
- Workerman 的
Worker::run()生命周期不接管协程调度,go()必须在你自己启动的协程环境里执行 - 若真要混用,建议明确分层:Workerman 做连接接入和路由,再把业务逻辑投递到 Swoole TaskWorker 或独立协程服务中处理
- 更稳妥的做法是二选一:小团队/快速迭代选 Workerman;有性能压测指标、IO 密集型服务、已有 Swoole 经验,直接上 Swoole
真正容易被忽略的,不是“哪个更快”,而是“谁承担调试成本”——协程让代码变线性,但也让问题更隐蔽:一次超时可能卡在 DNS、SSL 握手、甚至内核 socket buffer,而不是一眼可见的 while(true) 循环。











