0

0

如何在Golang中实现Server-Sent Events服务端推送 Go语言SSE实时数据流

P粉602998670

P粉602998670

发布时间:2026-03-11 12:50:53

|

966人浏览过

|

来源于php中文网

原创

go的http.responsewriter需保持长连接以支持sse:设text/event-stream类型、禁用缓存、手动flush、严格遵循event/data/id格式、每连接独立goroutine+带缓冲channel、检测客户端断连并加心跳重连。

如何在golang中实现server-sent events服务端推送 go语言sse实时数据流

Go 的 http.ResponseWriter 必须保持长连接不关闭

Server-Sent Events 本质是 HTTP 长连接流式响应,服务端不能写完就关连接。Go 默认在 handler 返回后自动关闭连接,必须手动干预。

常见错误现象:net/http: request method or response status code does not allow body 或客户端只收到一次数据就断连。

  • w.Header().Set("Content-Type", "text/event-stream") 声明 MIME 类型
  • 禁用缓冲:w.Header().Set("Cache-Control", "no-cache")w.Header().Set("Connection", "keep-alive")
  • 调用 w.(http.Flusher).Flush() 强制刷出响应头和每条事件,否则数据卡在缓冲区
  • 不要用 fmt.Fprintf 直接写入,改用 io.WriteString + Flush() 组合,避免隐式换行或编码干扰

event:data:id: 这些字段必须严格遵循 SSE 协议格式

浏览器的 EventSource 对解析非常敏感,任意拼写错误或空格缺失都会导致事件被静默丢弃。

使用场景:推送日志、实时计数、状态变更通知——所有需要按“事件类型+数据”区分处理的流式场景。

立即学习go语言免费学习笔记(深入)”;

皮卡智能
皮卡智能

AI驱动高效视觉设计平台

下载
  • 每条消息以空行分隔;data: 后必须跟一个换行,再写内容;多行 data: 要重复写前缀(data: line1\ndata: line2\n\n
  • event: 字段决定 addEventListener("xxx", ...) 中的事件名,不写则默认为 message
  • id: 用于断线重连时告诉服务端从哪条消息继续,但 Go 侧需自行维护序列号,协议本身不提供自动恢复
  • 别漏掉末尾的双换行:fmt.Fprint(w, "data: hello\n\n"); f.Flush()

并发推送时,每个连接要独立 goroutine + channel 控制

一个 handler 对应一个客户端连接,如果多个客户端共用同一个 channel,会互相覆盖或阻塞。

性能影响:goroutine 开销小(~2KB 栈),但 channel 缓冲区大小没设好会导致推送卡住或丢失。

  • 为每个连接启动独立 goroutine:go func(w http.ResponseWriter, r *http.Request) { ... }(w, r)
  • 用带缓冲的 chan string(如 make(chan string, 16))接收业务层推送,防止生产者阻塞
  • 在 goroutine 内循环读 channel,每次写完都 Flush();channel 关闭时主动 break 并 return
  • 注意:不要在 handler 里直接 time.Sleep 模拟推送,会阻塞整个 HTTP 处理器

客户端断连后,Go 服务端不会立刻感知 write 失败

HTTP 连接断开后,w.Write() 可能仍成功返回(TCP FIN 还没传回),直到下次写或 flush 才报错,容易堆积 goroutine。

容易踩的坑:用 select 等待 channel + context.Done() 是不够的,必须检查底层连接状态。

  • 每次 Flush() 后检查:if f, ok := w.(http.Flusher); ok && !isClientConnected(r.Context()) { return }
  • 判断连接是否存活:Go 1.21+ 可用 r.Context().Err() == context.Canceled,但更可靠的是捕获 write: broken pipeuse of closed network connection 错误
  • http.ResponseWriter 包装成支持中断的 writer,或用 net.Conn.SetWriteDeadline 配合超时检测
  • 别依赖 defer 清理资源——goroutine 可能永远卡在 channel receive 上

真正难处理的是网络闪断和 NAT 超时,SSE 没有心跳机制,得自己加 retry: 和定时 ping 事件,否则用户切后台几分钟后回来就收不到更新了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

247

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

356

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

214

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

409

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

490

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

200

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

1438

2025.06.17

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 6.1万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号