PHP不能直接作为WebSocket客户端或服务端实现实时推送,需借助Swoole/Workerman等独立服务,PHP仅通过HTTP/Redis等方式通知其转发消息。

PHP 本身不支持直接作为 WebSocket 客户端长期维持连接并主动推送消息到浏览器(比如用 fsockopen 或 stream_socket_client 建连后发一次就断,无法“实时推送”),更不是 WebSocket 服务端。想实现“PHP 连接 WebSocket 实时推送”,实际要分清角色:PHP 是消息源(比如订单生成、日志触发),真正负责长连接和广播的是独立的 WebSocket 服务;PHP 只需把数据“通知”过去。
PHP 怎么把数据发给 WebSocket 服务(如 Swoole / Workerman)
典型做法是让 PHP 脚本通过 HTTP、Redis 或 Socket 向已运行的 WebSocket 服务端发送指令,由它转发给在线客户端。例如:
- WebSocket 服务(如 Workerman)监听
http://127.0.0.1:2345接收推送请求,PHP 用file_get_contents或cURLPOST 数据过去 - 或统一写入 Redis 的
publish频道,WebSocket 进程订阅该频道,收到后调用$connection->send() - 避免用
socket_write直连 WebSocket 服务的 ws 端口——那需要手动处理 WebSocket 帧格式(MASK、FIN、opcode),极易出错且无必要
为什么不能用 curl_init("ws://...") 或 new WebSocket() 在 PHP 里?
因为 PHP 标准库完全不支持 WebSocket 协议握手与帧解析;curl_init 不识别 ws:// 协议(会报 Unsupported protocol);JS 的 WebSocket 构造函数只在浏览器环境有效,PHP 中不存在。常见错误现象包括:
-
PHP Warning: curl_setopt(): Invalid curl configuration option(设了CURLOPT_URL为ws://) -
Fatal error: Class 'WebSocket' not found(误以为 PHP 内置该类) - 用
stream_socket_client("tcp://...")连上端口后发原始字符串,但浏览器收不到——缺 WebSocket 握手响应(HTTP Upgrade)和帧封装
推荐最小可行组合:PHP + Workerman + Redis
Workerman 自带 WebServer 和 WebsocketConnection,配合 Redis pub/sub 实现解耦:
立即学习“PHP免费学习笔记(深入)”;
- 启动 Workerman WebSocket 服务,监听
0.0.0.0:2346,同时redis->subscribe(['notify_channel']) - PHP 业务逻辑中执行:
$redis->publish('notify_channel', json_encode(['event'=>'order_created','data'=>$order])) - Workerman 收到后遍历所有
$connections,对每个调用$connection->send(...) -
前端 JS 用标准
new WebSocket("ws://your-domain.com:2346")接收
注意:Workerman 进程必须常驻(php start.php start -d),不能跑在 Apache/Nginx 的 CGI 模式下——那每次 HTTP 请求都重启进程,连接立刻断开。
如果只是临时测试,用 text/plain HTTP 推送替代 WebSocket?
某些场景(如后台任务完成通知管理员),不需要毫秒级实时,可用更轻量方案:
- 前端用
EventSource订阅 PHP 输出的流:new EventSource("/api/sse.php") - PHP 脚本中设置
header('Content-Type: text/event-stream'),循环echo "data: ...\n\n"; ob_flush(); flush(); sleep(1); - PHP 业务触发时,往共享存储(如文件、Redis)写状态,SSE 脚本轮询读取并推送给客户端
这种方案绕开了 WebSocket 复杂性,兼容性更好(IE 不支持,但现代浏览器都行),也更容易调试——但别把它当成“WebSocket 推送”的等价实现,延迟和连接模型完全不同。
真正卡住人的往往不是代码怎么写,而是没意识到 PHP 进程生命周期和 WebSocket 长连接本质冲突;只要把“PHP 发指令”和“WebSocket 服务收指令并广播”拆成两个独立环节,问题就清晰了。











