当前位置: 首页 > news >正文

GO GIN SSE DEMO

文章目录

      • 接口描述:
          • 1.1 `/events/time` - 时间流
          • 1.2 `/events/numbers` - 数字流
        • 2. 用户管理接口
          • 2.1 `/user/:id` - 获取用户信息
          • 2.2 `/user` - 创建用户
      • 项目结构
        • 1. `main.go`
        • 2. 创建 `handlers/event_time.go`
        • 3. 创建 `handlers/event_number.go`
        • 4. `handlers/user.go`
        • 5. 运行服务器
      • 小结

将GIN项目与SSE库结合,写一个简单的demo,具备正常的API接口和流处理接口,实现了两种类型的 Server-Sent Events (SSE) 流,以及一个简单的用户管理接口。

下面是对每个接口的详细描述:

接口描述:

####1. SSE 接口

1.1 /events/time - 时间流
  • 功能:每秒向客户端发送一次当前的时间。
  • 路径/events/time
  • 请求方式GET
  • 返回格式:SSE(文本事件流 text/event-stream
  • 数据内容:每条消息包含当前时间,格式为 RFC3339,例如:data: 2024-09-14T15:00:00Z
  • 实现细节
    • 使用 Gin 处理 HTTP 请求。
    • 设置 HTTP 头以指示客户端进行流式传输。
    • 通过 for 循环持续发送当前时间,并使用 flusher.Flush() 刷新输出缓冲区,确保数据实时传送到客户端。
    • 每次发送时间后,等待 1 秒钟。
1.2 /events/numbers - 数字流
  • 功能:每 2 秒向客户端发送一个递增的数字。
  • 路径/events/numbers
  • 请求方式GET
  • 返回格式:SSE(文本事件流 text/event-stream
  • 数据内容:每条消息包含一个递增的数字,例如:data: 0data: 1data: 2 等。
  • 实现细节
    • 与时间流相似,设置 HTTP 头并初始化 SSE 传输。
    • 使用 for 循环持续发送递增的数字,每发送一次数字后,等待 2 秒钟。
2. 用户管理接口
2.1 /user/:id - 获取用户信息
  • 功能:根据用户 ID 获取用户信息。
  • 路径/user/:id
  • 请求方式GET
  • 参数
    • id:用户的唯一标识符,通过 URL 参数传递。
  • 返回格式:JSON
  • 数据内容:返回包含用户 ID 和名称的 JSON 对象,例如:{"user_id": "1", "name": "John Doe"}
  • 实现细节
    • 从 URL 中获取用户 ID。
    • 假设从数据库或其他数据源获取用户信息,当前示例中返回了硬编码的用户信息。
2.2 /user - 创建用户
  • 功能:创建一个新的用户。
  • 路径/user
  • 请求方式POST
  • 请求体:JSON 格式,包含用户名称,例如:{"name": "Alice"}
  • 返回格式:JSON
  • 数据内容:返回新创建的用户信息,包括用户 ID 和名称,例如:{"user_id": "12345", "name": "Alice"}
  • 实现细节
    • 解析请求体中的 JSON 数据,获取用户名称。
    • 验证数据是否合法。
    • 假设将用户信息存储到数据库或其他存储系统,当前示例中返回了硬编码的用户 ID 和名称。

项目结构

project/
│
├── main.go
├── handlers/
│   ├── event_number.go
│   ├── event_time.go
│   ├── user.go
1. main.go

main.go 文件会引入所有的路由注册函数,包括 event_timeevent_number

package mainimport ("project/handlers""github.com/gin-gonic/gin"
)func main() {router := gin.Default()// 注册 SSE 路由handlers.RegisterEventTimeRoutes(router)handlers.RegisterEventNumberRoutes(router)// 注册用户路由handlers.RegisterUserRoutes(router)router.Run(":8080")
}
2. 创建 handlers/event_time.go

event_time.go 文件只处理时间流的 SSE 接口。

package handlersimport ("fmt""net/http""time""github.com/gin-gonic/gin"
)func RegisterEventTimeRoutes(router *gin.Engine) {router.GET("/events/time", timeStream)
}// 时间流
func timeStream(c *gin.Context) {c.Writer.Header().Set("Content-Type", "text/event-stream")c.Writer.Header().Set("Cache-Control", "no-cache")c.Writer.Header().Set("Connection", "keep-alive")flusher, ok := c.Writer.(http.Flusher)if !ok {c.String(http.StatusInternalServerError, "Streaming unsupported!")return}for {// 发送当前时间fmt.Fprintf(c.Writer, "data: %s\n\n", time.Now().Format(time.RFC3339))flusher.Flush()time.Sleep(1 * time.Second)}
}
3. 创建 handlers/event_number.go

event_number.go 文件只处理数字流的 SSE 接口。

package handlersimport ("fmt""net/http""time""github.com/gin-gonic/gin"
)func RegisterEventNumberRoutes(router *gin.Engine) {router.GET("/events/numbers", numbersStream)
}// 数字流
func numbersStream(c *gin.Context) {c.Writer.Header().Set("Content-Type", "text/event-stream")c.Writer.Header().Set("Cache-Control", "no-cache")c.Writer.Header().Set("Connection", "keep-alive")flusher, ok := c.Writer.(http.Flusher)if !ok {c.String(http.StatusInternalServerError, "Streaming unsupported!")return}number := 0for {// 发送递增的数字fmt.Fprintf(c.Writer, "data: %d\n\n", number)flusher.Flush()number++time.Sleep(2 * time.Second)}
}
4. handlers/user.go

user.go 文件保持不变,用于处理用户相关的接口。

package handlersimport ("net/http""github.com/gin-gonic/gin"
)func RegisterUserRoutes(router *gin.Engine) {router.GET("/user/:id", getUser)router.POST("/user", createUser)
}func getUser(c *gin.Context) {userId := c.Param("id")// 假设在这里处理获取用户信息的逻辑c.JSON(http.StatusOK, gin.H{"user_id": userId,"name":    "John Doe",})
}func createUser(c *gin.Context) {var json struct {Name string `json:"name" binding:"required"`}if err := c.ShouldBindJSON(&json); err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})return}// 假设在这里处理创建用户的逻辑c.JSON(http.StatusCreated, gin.H{"user_id": "12345","name":    json.Name,})
}
5. 运行服务器
  1. main.goevent_time.goevent_number.gouser.go 保存到项目中。

  2. 运行项目:

    go run main.go
    
  3. 服务器将在 localhost:8080 上运行。

小结

  • 模块化:将每个 SSE 接口拆分到单独的文件中,可以提高代码的可读性和可维护性。
  • 路由注册:在 main.go 中分别调用 RegisterEventTimeRoutesRegisterEventNumberRoutes 注册各自的路由。
  • 灵活扩展:这种结构可以很容易地扩展新的 SSE 接口,只需创建新的文件并注册路由。

这样做可以让每个文件更专注于自己的任务,保持代码清晰整洁,有利于团队合作和代码维护。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 【Flink实战】flink消费http数据并将数组展开多行
  • AI健身之俯卧撑计数和姿态矫正-角度估计
  • 【JavaEE初阶】多线程7(面试要点)
  • 亚马逊IP关联揭秘:发生ip关联如何处理
  • [Python学习日记-27] 文件操作练习题解析
  • python新手的五个练习题
  • 计算机毕业设计 基于SpringBoot的小区运动中心预约管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解
  • SpringBoot+Vue技术框架开发的ADR智能监测系统源码,Java语言的药品不良反应智能监测系统源代码
  • Vue工程师面试题
  • Homebrew安装与切换下载源
  • vue3中把封装svg图标为全局组件
  • 解决SVN蓝色问号的问题
  • Android开发高频面试题之——kotlin篇
  • 网络层协议 —— IP协议
  • MyBatis 源码解析:TypeHandler 设计与自定义实现
  • 《Java编程思想》读书笔记-对象导论
  • CNN 在图像分割中的简史:从 R-CNN 到 Mask R-CNN
  • Cookie 在前端中的实践
  • css系列之关于字体的事
  • ES6语法详解(一)
  • hadoop集群管理系统搭建规划说明
  • JavaScript异步流程控制的前世今生
  • java小心机(3)| 浅析finalize()
  • leetcode-27. Remove Element
  • miaov-React 最佳入门
  • PHP 小技巧
  • SpiderData 2019年2月25日 DApp数据排行榜
  • vue自定义指令实现v-tap插件
  • 从零开始在ubuntu上搭建node开发环境
  • 给新手的新浪微博 SDK 集成教程【一】
  • 构造函数(constructor)与原型链(prototype)关系
  • 机器人定位导航技术 激光SLAM与视觉SLAM谁更胜一筹?
  • 模型微调
  • 前端每日实战:70# 视频演示如何用纯 CSS 创作一只徘徊的果冻怪兽
  • 使用 QuickBI 搭建酷炫可视化分析
  • 掌握面试——弹出框的实现(一道题中包含布局/js设计模式)
  • 主流的CSS水平和垂直居中技术大全
  • 继 XDL 之后,阿里妈妈开源大规模分布式图表征学习框架 Euler ...
  • #pragma multi_compile #pragma shader_feature
  • #pragma once
  • $L^p$ 调和函数恒为零
  • ( 10 )MySQL中的外键
  • (14)目标检测_SSD训练代码基于pytorch搭建代码
  • (152)时序收敛--->(02)时序收敛二
  • (2/2) 为了理解 UWP 的启动流程,我从零开始创建了一个 UWP 程序
  • (ISPRS,2021)具有遥感知识图谱的鲁棒深度对齐网络用于零样本和广义零样本遥感图像场景分类
  • (LeetCode 49)Anagrams
  • (ResultSet.TYPE_SCROLL_INSENSITIVE,ResultSet.CONCUR_READ_ONLY)讲解
  • (WSI分类)WSI分类文献小综述 2024
  • (八十八)VFL语言初步 - 实现布局
  • (附源码)spring boot校园拼车微信小程序 毕业设计 091617
  • (附源码)计算机毕业设计SSM基于健身房管理系统
  • (附源码)计算机毕业设计大学生兼职系统
  • (九十四)函数和二维数组
  • (利用IDEA+Maven)定制属于自己的jar包