fsockopen 可连接 WebSocket 但需手动实现完整协议:先拼接含 Sec-WebSocket-Key 的 HTTP 握手请求并校验 Sec-WebSocket-Accept 响应,再按 RFC 6455 手动分帧(客户端消息必掩码)、处理跨包数据、超时重试等,否则易连而不用。

用 fsockopen 连 WebSocket 协议本身是可行的,但不能直接“连上就发消息”
WebSocket 是基于 HTTP 升级(Upgrade: websocket)的协议,fsockopen 只能建立原始 TCP 连接并发送/接收裸字节,它不解析 HTTP 头、不处理 WebSocket 帧格式(如掩码、opcode、长度编码)。所以你得自己拼接握手请求、校验响应头、再手动打包/解包数据帧——这不是“连接”,而是“从零实现客户端”。
- 握手阶段必须发送正确的
Sec-WebSocket-Key,服务端会用它生成Sec-WebSocket-Accept响应,少一个 header 或 base64 错一位都会失败 - 后续所有消息必须按 WebSocket 规范(RFC 6455)分帧:客户端发的消息必须带掩码(mask=1),服务端发的不能掩码;你漏掉
mask key或没异或 payload,对方直接断连 -
fsockopen没有超时重试、心跳保活、自动重连逻辑,出错后全靠你自己fgets/fwrite+feof+fclose手动兜底
实际项目中用 fsockopen 连 WebSocket 的典型错误现象
最常见的不是连不上,而是“连上了却收不到消息”或“发一条就断”。比如:
- 握手返回
HTTP/1.1 101 Switching Protocols,但缺少Sec-WebSocket-Accept或值校验失败 → 连接被服务端静默关闭 - 发消息时没设 mask 位(第 9 个 bit),或 mask key 四字节没参与 payload 异或 → 服务端按协议直接 close 连接,状态码常为
1002 - 读取响应时只用
fgets($fp),但 WebSocket 数据帧可能跨 TCP 包到达,fgets遇不到换行就阻塞或截断 → 后续帧解析全乱
比 fsockopen 更靠谱的替代方案
除非你在嵌入式环境或极度受限的 PHP 环境(连 ext-sockets 都没编译),否则别硬刚 fsockopen。推荐路径:
- 用
ext-sockets+ 自己写帧逻辑:比fsockopen更底层可控,支持非阻塞、select 轮询,适合长连接管理 - 用
reactphp/socket+textalk/websocket:Composer 安装即用,自动处理握手、ping/pong、分帧、重连,$conn->send()和$conn->on('message', ...)就完事 - 如果只是偶尔发指令(比如通知前端刷新),改用服务端 HTTP 接口 + 前端 WebSocket 主动轮询或监听事件,绕开 PHP 端维持 WebSocket 连接的复杂度
真要用 fsockopen,关键步骤不能跳
最小可运行握手示例的核心环节(省略错误处理):
立即学习“PHP免费学习笔记(深入)”;
$host = 'echo.websocket.org';
$port = 80;
$fp = fsockopen($host, $port, $errno, $errstr, 5);
if (!$fp) die("connect failed: $errstr ($errno)");
// 生成 Sec-WebSocket-Key(必须是 16 字节随机 base64)
$key = base64_encode(random_bytes(16));
// 发送握手请求
$request = "GET / HTTP/1.1\r\n";
$request .= "Host: $host\r\n";
$request .= "Upgrade: websocket\r\n";
$request .= "Connection: Upgrade\r\n";
$request .= "Sec-WebSocket-Key: $key\r\n";
$request .= "Sec-WebSocket-Version: 13\r\n\r\n";
fwrite($fp, $request);
// 读响应头(直到遇到空行)
$response = '';
while (!feof($fp) && ($line = fgets($fp)) !== false) {
$response .= $line;
if ($line === "\r\n") break;
}
// 校验 HTTP 状态和 Accept 头
if (strpos($response, "HTTP/1.1 101") === false) {
die("handshake failed");
}
$accept = 'Sec-WebSocket-Accept: ' . base64_encode(
sha1($key . '258EAFA5-E914-47DA-95CA-C5AB0DC85B11', true)
);
if (strpos($response, $accept) === false) {
die("accept mismatch");
}
到这里才算“握手成功”。之后所有通信都要按 RFC 6455 手动构造帧——这才是真正容易翻车的地方,比如掩码 key 必须每次随机、payload 长度超过 125 要扩展 length 字段、控制帧(ping)必须原样 echo 回去……这些细节一旦出错,连接立刻中断,且很难定位。










