

新闻资讯
技术学院Go中WebSocket心跳需服务端启用PingHandler并每25秒发ping、客户端每30秒发JSON心跳包,双方均需超时检测(服务端45秒、客户端60秒)并主动断连,同时注意Nginx超时配置与WriteMessage并发安全。
在 Go 中实现 WebSocket 心跳机制,核心是客户端与服务端协同发送 ping/pong 帧或自定义心跳消息,防止连接因中间代理(如 Nginx、负载均衡器)超时断开。标准 WebSocket 协议本身支持 Ping/Pong 控制帧,但很多 Go 的 WebSocket 库(如 gorilla/websocket)默认不自动响应 pong,需手动处理;同时,应用层心跳(如 JSON 消息)更可控、可监控,推荐两者结合使用。
使用 gorilla/websocket 时,需显式设置 CheckOr、启用
iginSetPingHandler 并启动协程定期写入 ping:
conn.SetPingHandler 注册 pong 响应逻辑(收到 ping 自动回 pong)time.Ticker 每 25 秒调用 conn.WriteMessage(websocket.PingMessage, nil)
conn.SetReadDeadline),并在读循环中捕获 websocket.CloseMessage 或超时错误及时关闭连接浏览器原生 WebSocket 会自动响应服务端 ping,无需额外处理;但为兼容性及可观测性,建议在 JS 客户端也定时发送 JSON 格式的心跳包(如 {"type":"heartbeat"}):
setInterval 每 30 秒 ws.send(JSON.stringify({type:'heartbeat'}))
message 事件,忽略服务端返回的纯 pong,只处理业务消息和自定义心跳响应onclose / onerror)触发重连逻辑,避免假死仅发心跳不够,必须配合超时检测才能真正识别“假在线”:
conn.WriteMessage(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, "")) 关闭实际部署中容易忽略以下细节:
proxy_read_timeout 是 60 秒,需设为大于服务端心跳间隔(如 75s),并配置 proxy_set_header Connection '' 避免连接被复用干扰WriteMessage 不是并发安全的,多个 goroutine 写需加锁或用 conn.WriteJSON + 单写协程通道模型SetPongHandler 来做业务逻辑——它只用于维持链路,业务心跳应走文本消息通道,便于日志追踪和协议扩展