Workerman内存持续上涨而memory_get_usage()无明显变化,主因是PHP引用计数未及时回收,尤其长连接下静态变量、闭包捕获、循环引用导致逻辑死亡对象未释放;需手动gc_collect_cycles()、用WeakMap替代静态数组、避免全局单例、清理onWorkerStop资源,并借助Xdebug分析引用图。

Workerman 进程内存持续上涨,memory_get_usage() 却没明显变化?
这大概率是 PHP 的引用计数机制没及时回收对象,尤其在长连接场景下。Workerman 不会自动 GC 静态变量或闭包捕获的上下文,memory_get_usage() 只统计当前脚本堆内存,不包含 PHP 内部引用计数未清但实际已“逻辑死亡”的对象——这类泄漏得靠 gc_collect_cycles() + memory_get_peak_usage() 对比才能暴露。
- 每轮业务逻辑结束后手动调用
gc_collect_cycles(),并在关键节点打点记录memory_get_peak_usage(true) - 避免在
onMessage或定时器回调里直接赋值给self::$cache这类静态属性,除非你明确控制其生命周期 - 用
xdebug_debug_zval()(需开启 Xdebug)检查可疑对象的 refcount 和 is_ref,确认是否被意外持有了引用
Worker::runAll() 启动后,static $instances = [] 越来越大?
Workerman 的每个子进程独立运行,静态变量不会跨进程共享,但会在单个进程生命周期内持续累积。如果在 onConnect 里往静态数组塞连接实例,又没在 onClose 清理,就会形成典型的循环引用泄漏:连接对象 → 回调闭包 → 持有当前类实例 → 又持有该静态数组。
- 改用弱引用容器:用
WeakMap(PHP 8.0+)替代static array存储连接上下文 - 若必须用静态数组,确保
onClose中执行unset(self::$instances[$connection->id]),且不要用$connection本身作键(它可能被其他地方强引用) - 禁用
gc_disable()—— Workerman 默认不关 GC,但某些扩展或自定义启动脚本可能误关,检查入口文件是否有这行
用了 SwooleTable 或 Redis 连接池,内存还是涨?
Workerman 本身不提供连接池,很多人混用 Swoole 组件时忽略进程模型差异:SwooleTable 是进程级共享内存,但 Workerman 子进程各自初始化一份;Redis 连接若在 Worker 构造函数里 new 出来,也会随进程常驻,且连接对象内部可能持有回调、超时定时器等隐式引用。
- SwooleTable 必须在
Worker::onWorkerStart中创建,且仅在该回调内使用global $table,别在onMessage里重复 new - Redis 实例建议用懒加载 + 静态属性,但每次 get 前判断
isConnected(),断连后显式unset($this->redis),防止重连时旧实例残留 - 所有第三方 SDK(如 EasyWeChat、Alibaba Cloud SDK)尽量避免全局单例,它们内部常带静态事件监听器或缓存,优先用工厂模式按需生成
ERROR: fork() failed, errno=12 出现前几小时内存就卡在 1.2G 上不去?
这不是 OOM,是 Linux fork() 失败的典型信号:父进程内存太大,子进程 copy-on-write 时预分配失败。Workerman 在 reload 或平滑重启时会 fork 新进程,此时老进程若已积累大量未释放对象(比如日志缓冲区、解析过的 XML DOM 树、大数组缓存),就会触发这个错误。
- 在
Worker::onWorkerStop中主动清理:unset($GLOBALS['my_big_cache'])、Logger::flush()、libxml_clear_errors() - 禁用 PHP 的
opcache.save_comments=1(尤其 Laravel 类框架大量注解时),它会让 opcache 内存占用翻倍且难回收 - 用
ps aux --sort=-%mem | head -20定期观察 Workerman 子进程 RSS,RSS 持续 >1G 就要查strace -p PID -e trace=brk,mmap,munmap看系统调用异常
真正麻烦的不是泄漏本身,而是泄漏路径藏在层层闭包和魔术方法里——__destruct 不一定被调,__sleep 可能悄悄复制了整个对象图。上线前用 php -d xdebug.mode=develop your_worker.php 开启 Xdebug 的引用图分析,比盲猜快得多。










