[转]GIN框架中间件 C.NEXT() C.ABORT() C.SET() C.GET 跨中间件取值 GIN中间件中使用GOROUTINE
-------------
1. C.NEXT()演示
//HandlerFunc
func indexHandler(c *gin.Context) {
fmt.Println("index")
c.JSON(http.StatusOK, gin.H{
"msg": "index",
})
}
//定义一个中间件
func m1(c *gin.Context) {
fmt.Println("m1 in ...")
// 计时
start := time.Now()
c.Next() //调用后续的处理函数
//c.Abort() //阻止调用后续的处理函数
cost := time.Since(start)
fmt.Printf("cost:%v\n", cost)
fmt.Printf("m1 out ...")
}
func m2(c *gin.Context) {
fmt.Println("m2 in ...")
c.Next() //调用后续的处理函数
fmt.Println("m2 out ...")
}
func main() {
r := gin.Default()
//点GET查看源码发现
//GET(relativePath string, handlers ...HandlerFunc) IRoutes 另外,...表示 可传多个HandlerFunc类型的函数
//r.GET("/index", m1, indexHandler)
//r.GET("/shop", m1, func(c *gin.Context) {
// c.JSON(http.StatusOK, gin.H{
// "msg": "shop",
// })
//})
//r.GET("/user", m1, func(c *gin.Context) {
// c.JSON(http.StatusOK, gin.H{
// "msg": "user",
// })
//})
// 点开Use 查看源码发现 Use(middleware ...HandlerFunc) IRoutes
r.Use(m1, m2) //全局注册中间件函数m1 m2
/*访问/index
执行indexHandler之前 去执行注册的中间件 总的执行打印顺序是:m1 in -> m2 in -> index -> m2 out -> m1 out
*/
r.GET("/index", indexHandler)
r.GET("/shop", func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{
"msg": "shop",
})
})
r.GET("/user", func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{
"msg": "user",
})
})
r.Run(":9090")
}
执行结果:
执行逻辑:
代码执行图解:
2. C.ABORT()演示:
上面代码把m2这里改成阻止,其他地方不变,如下图:
此时执行,控制台打印输出为:
m1 in -> m2 in ->m2 out -> m1 out
- 1
代码执行图解:
当然,如果连 m2 out 都不想让它输出的话,那直接在c.Abort() 语句 后return就可以了,如下图:
3. C.SET() C.GET 跨中间件取值
//HandlerFunc
func indexHandler(c *gin.Context) {
fmt.Println("index")
name, ok := c.Get("name") //取值 实现了跨中间件取值
if !ok{
name = "default user"
}
c.JSON(http.StatusOK, gin.H{
"msg": name,
})
}
//定义一个中间件
func m1(c *gin.Context) {
fmt.Println("m1 in ...")
// 计时
start := time.Now()
//gin中间件中使用goroutine
//当在中间件或handler中启动新的goroutine时,不能使用原始的上下文(c *gin.Context),必须使用其只读副本(c.Copy())
go otherFunc(c.Copy()) // 在otherFunc中只能使用c的拷贝
c.Next() //调用后续的处理函数
//c.Abort() //阻止调用后续的处理函数
cost := time.Since(start)
fmt.Printf("cost:%v\n", cost)
fmt.Printf("m1 out ...")
}
func m2(c *gin.Context) {
fmt.Println("m2 in ...")
c.Set("name", "tony") //可以在请求上下文里面设置一些值,然后其他地方取值
c.Abort() //阻止调用后续的处理函数 也就是 m2它自己走完就行
//return
fmt.Println("m2 out ...")
}
// 自定义认证中间件 通过这种方式实现一些灵活的控制
func authMiddleware(doCheck bool)gin.HandlerFunc {
//连接数据库
// 或其他一些准备工作
return func(c *gin.Context) {
if doCheck {
//这里存放具体的逻辑
// 是否登录的判断
// if 是登录用户
c.Next()
// else
// c.Abort()
}else {
c.Next()
}
}
}
func main() {
r := gin.Default()
// 点开Use 查看源码发现 Use(middleware ...HandlerFunc) IRoutes
r.Use(m1, m2, authMiddleware(true)) //全局注册中间件函数m1 m2 authMiddleware
/*访问/index
执行indexHandler之前 去执行注册的中间件 总的执行打印顺序是:m1 in -> m2 in -> index -> m2 out -> m1 out
*/
r.GET("/index", indexHandler)
r.GET("/shop", func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{
"msg": "shop",
})
})
r.GET("/user", func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{
"msg": "user",
})
})
/* 路由组注册中间件方法1
routeGroup1 := r.Group("/xx", authMiddleware(true)){
routeGroup1.GET("/index", func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{"msg": "routeGroup1"})
})
}
路由组注册中间件方法2
routeGroup2 := r.Group("/xx2")
routeGroup2.Use(authMiddleware(true)){
routeGroup2.GET("/index", func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{"msg": "routeGroup2"})
})
}
*/
r.Run(":9090")
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 25岁的心里话
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
2019-10-10 Multiple inheritance in Go
2019-10-10 Inheritance and subclassing in Go - or its near likeness
2019-10-10 golang Methods on structs
2018-10-10 【转】6 Reasons Why JavaScript’s Async/Await Blows Promises Away (Tutorial)
2017-10-10 【转】shell中的内建命令, 函数和外部命令
2017-10-10 clear out all variables without closing terminal
2017-10-10 linux中shell命令test用法和举例