Golang中实时推送的功臣 - WebSocket

Golang中实时推送的功臣 - WebSocket

HTTP的问题是 一问一答,短连接

  • 客户端主动发请求 → 服务端才返回数据
  • 服务端不能主动给客户端发消息
  • 每次通信都要重新建立连接(像发短信

WebSocket 就是为了解决这个问题诞生的,基于 TCP 的 双向长连接 通信协议

  • 一次连接,永久保持(直到断开)
  • 客户端 ↔ 服务端 随时互相发消息
  • 服务端可以主动推送消息给客户端
  • 打电话:接通后,你随时说,对方随时说

WebSocket 是一个「实时双向通信」的协议,底层用 TCP,借用 HTTP 建立连接,专门用来做实时功能。

WebSocket 能干嘛?

  • 服务端主动推消息(HTTP 做不到!)

  • 双向实时通信(毫秒级延迟)

  • 长连接,不用反复创建连接

  • 轻量,开销比 HTTP 小很多

  • 浏览器原生支持(网页直接用)

  • 实时推送、聊天、直播

极简示例

go 复制代码
package main

import (
	"fmt"
	"net/http"

	"github.com/gorilla/websocket" // go get github.com/gorilla/websocket
)

// 定义 WebSocket 升级器:把 HTTP 连接升级成 WebSocket
var upgrader = websocket.Upgrader{
	// 允许跨域(测试用)
	CheckOrigin: func(r *http.Request) bool {
		return true
	},
}

// 处理 WebSocket 连接
func wsHandler(w http.ResponseWriter, r *http.Request) {
	// 1. 把 HTTP 连接升级为 WebSocket 连接
	conn, err := upgrader.Upgrade(w, r, nil)
	if err != nil {
		fmt.Println("升级失败:", err)
		return
	}
	defer conn.Close()

	fmt.Println("客户端已连接!")

	// 2. 循环接收/发送消息(长连接)
	for {
		// 读取客户端消息
		msgType, msg, err := conn.ReadMessage()
		if err != nil {
			fmt.Println("客户端断开连接")
			break
		}

		fmt.Printf("收到消息:%s\n", string(msg))

		// 服务端主动回消息
		reply := "服务端收到:" + string(msg)
		conn.WriteMessage(msgType, []byte(reply))
	}
}

func main() {
	http.HandleFunc("/ws", wsHandler)
	fmt.Println("WebSocket 服务启动:ws://127.0.0.1:8081/ws")
	http.ListenAndServe(":8081", nil)
	// 测试: wscat -c ws://127.0.0.1:8081/ws
}

核心 API

websocket.Upgrader把 HTTP 连接 → 升级成 WebSocket 长连接(WebSocket 必须先用 HTTP 握手)
go 复制代码
var upgrader = websocket.Upgrader{
	CheckOrigin: func(r *http.Request) bool {
		return true // 允许跨域(开发必开)
	},
}
upgrader.Upgrade()执行升级,得到 WebSocket 连接对象 *websocket.Conn
go 复制代码
conn, err := upgrader.Upgrade(w, r, nil)
conn.ReadMessage()阻塞读取客户端发来的消息返回:消息类型、消息内容、错误
go 复制代码
msgType, msg, err := conn.ReadMessage()
conn.WriteMessage()服务端主动发消息给客户端(核心能力!)
go 复制代码
conn.WriteMessage(msgType, []byte("你好客户端"))
conn.ReadJSON() / conn.WriteJSON()直接收发 JSON 结构体(不用手动序列化)
go 复制代码
// 接收 JSON
var data map[string]any
conn.ReadJSON(&data)

// 发送 JSON
conn.WriteJSON(map[string]any{
    "code": 200,
    "msg":  "服务端推送",
})
相关推荐
为何创造硅基生物2 小时前
C语言 结构体内存对齐规则(通俗易懂版)
c语言·开发语言
吃好睡好便好2 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
星寂樱易李2 小时前
iperf3 + Python-- 网络带宽、网速、网络稳定性
开发语言·网络·python
仰泳之鹅2 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
之歆2 小时前
DAY_12JavaScript DOM 完全指南(二):实战与性能篇
开发语言·前端·javascript·ecmascript
姚不倒3 小时前
Go语言进阶:接口、错误处理与并发编程(goroutine/channel/context)
云原生·golang
candyTong3 小时前
Claude Code 的 Edit 工具是怎么工作的
javascript·后端·架构
cen__y3 小时前
Linux12(Git01)
linux·运维·服务器·c语言·开发语言·git
AI人工智能+电脑小能手3 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
GetcharZp4 小时前
GitHub 2.4 万 Star!D2 正在重新定义程序员画图方式
后端