感觉用不用效果一样的啊,看文档说是执行挂起程序,具体怎么好理解一些呢,每个中间件一定要用吗?
package main
import (
"fmt"
"github.com/gin-gonic/gin"
)
func main() {
r := gin.Default()
r.GET("/ping", Ping)
r.Use(PrintRequest)
r.Use(PrintResponse)
// add your handlers here
r.Run()
}
func PrintRequest(c *gin.Context) {
fmt.Println("this is request")
}
func PrintResponse(c *gin.Context) {
now := time.Now()
// 先执行handler下面中间件和handler
c.Next()
fmt.Printf("this is response, cost: %d",
time.Since(now).Nanoseconds())
}
我们以上面的代码为例,加入了两个中间件 PrintRequest
PrintResponse
PrintResponse
这个中间件里,进行计时就ok了。 当你打印返回的response body的时候,也是一样的道理,中间件都会比handler先执行,但是没有handler,中间件怎么拿到 response body,这时候就要用到 c.Next() 先去 执行 余下的中间件和 handler,然后再回到我这个中间件里面c.Next 的主要作用大概就是这样了
中间件用的,不要管啥挂不挂起的,我估计是翻译的锅。
假如你用两个中间件
// 打印请求处理事件
func Middleware1(ctx *gin.Context) {
start:=time.Now() // 记录开始时间
ctx.Next() // 调用处理过程(会产生调用耗时)
t:=time.Since(start) // 调用处理过程完毕后计算时间差
fmt.Println(t) // 打印本次请求处理时间差
}
func Middleware2(ctx *gin.Context) {
if(!servic e.CheckLogin(ctx)) { // 登录检测,未登录
return // 直接return,该请求的处理结束
}
ctx.Next() // 登录检测通过,继续后续处理
}
c.Next() 是让调该Handler 执行下一个Handler. 否则下一个Handler就执行不了。
gin.Get("/",func (ctx *gin.Context){
fmt.Println(3)
// 这个是最后一个,因此就不必要调用 ctx.Next() 了,这里是我们常用的控制器方法
}, func(ctx *gin.Context){
fmt.Println(2)
ctx.Next()
// 这里是我们的中间件
}, func(ctx *gin.Context){
fmt.Println(1)
ctx.Next()
})
我这里传了3个handler. 顺序传入,但是会逆序执行。
中间件的函数签名和执行控制器方法的签名一样,只是中间件一般都在控制器之前执行,所以叫中间件。。。
你写个类似用户认证的中间件就懂了,会用到c.Next
c.Abort
,你再看看这些代码就懂了,单纯在中间件打印一些信息用不用是无所谓的,因为本来中间件就会一层一层执行
可以看看源码中的解释,这个方法只在中间件中使用有意义,调用后马上执行各个 handler(这些 handler 中包含你注册在路由上的方法),最后会返回到中间件。
// Next should be used only inside middleware.
// It executes the pending handlers in the chain inside the calling handler.
// See example in GitHub.
func (c *Context) Next() {
c.index++
for s := int8(len(c.handlers)); c.index < s; c.index++ {
c.handlers[c.index](c)
}
}
理解是加上c.Next()后马上开始执行下一个中间件,同时也执行c.Next()后的代码。不加上c.Next()则要等自身的handler执行完成才开始下一个中间件。下面是对比代码输出:
func testCNext() {
r := gin.New()
mid1 := func(c *gin.Context) {
start := time.Now()
fmt.Println("middleware1 start")
// 注释 or 不注释
c.Next()
fmt.Println(time.Since(start))
fmt.Println("middleware1 ending")
}
mid2 := func(c *gin.Context) {
fmt.Println("middleware2 start")
c.Next()
fmt.Println("middleware2 ending")
}
r.Use(mid1, mid2)
r.GET("/", func(c *gin.Context) {
c.JSON(http.StatusOK, "hi")
})
r.Run()
}
不注释mid1中的c.Next()输出
middleware1 start
middleware2 start
middleware2 ending
175.819µs
middleware1 ending
注释mid1中的c.Next()输出
middleware1 start
13.139µs
middleware1 ending
middleware2 start
middleware2 ending
这个看下代码就很好理解了handleHTTPRequest
(1)Next在Gin服务启后,相关请求进来时就会调用,用于执行为这个路由注册的所有中间件,正常情况下就是每个中间件都会顺序执行完成,所以就能解答你的疑问,为啥中间件不加Next也会执行下去。Next函数只是遍历执行所有handler而已,没啥
(2)在中间件中调用Next是啥效果?比如handlerA中调用了Next,那么在A中执行到Next时,Next函数会遍历执行余下的handler,完成后,Next返回到A中,继续执行完A
5 回答5.7k 阅读✓ 已解决
6 回答3.4k 阅读
2 回答3.3k 阅读✓ 已解决
1 回答5.4k 阅读✓ 已解决
2 回答3.2k 阅读✓ 已解决
1 回答5.1k 阅读✓ 已解决
1 回答1.6k 阅读✓ 已解决
中间件可以理解为洋葱穿透。
c.Next() 之前的操作是在 Handler 执行之前就执行;
c.Next() 之后的操作是在 Handler 执行之后再执行;
然后你在Handler中输出一些内容就能发现。
Hello Before
在你的Handler之前就输出。Hello After
在之后输出。之前
的操作一般用来做验证处理,访问是否允许之类的。之后
的操作一般是用来做总结处理,比如格式化输出、响应结束时间,响应时长计算之类的。