2

在Gin框架中,中间件(Middleware)指的是可以拦截http请求-响应生命周期的特殊函数,在请求-响应生命周期中可以注册多个中间件,每个中间件执行不同的功能,一个中间执行完再轮到下一个中间件执行。

中间件的常见应用场景如下:

  • 请求限速
  • api接口签名处理
  • 权限校验
  • ...
提示:如果你想拦截所有请求做一些事情都可以开发一个中间件函数去实现。

1.使用中间件

func main() {
    r := gin.New()

    // 通过use设置全局中间件

    // 设置日志中间件,主要用于打印请求日志
    r.Use(gin.Logger())

    // 设置Recovery中间件,主要用于拦截paic错误,不至于导致进程崩掉
    r.Use(gin.Recovery())

    // 忽略后面代码
}

2.自定义中间件

下面我们来看一下gin中的中间件

// Use attaches a global middleware to the router. ie. the middleware attached though Use() will be
// included in the handlers chain for every single request. Even 404, 405, static files...
// For example, this is the right place for a logger or error management middleware.
func (engine *Engine) Use(middleware ...HandlerFunc) IRoutes {
    engine.RouterGroup.Use(middleware...)
    engine.rebuild404Handlers()
    engine.rebuild405Handlers()
    return engine
}

通过gin的源码,我们发现,中间件是一个gin.HandlerFunc

只要我们自定义的一个中间件是返回gin.HandlerFunc的函数都可以自定义一个中间件。

package main
// 导入gin包
import (
"github.com/gin-gonic/gin"
    "log"
    "time"
)

// 自定义个日志中间件
func Logger() gin.HandlerFunc {
    return func(c *gin.Context) {
        t := time.Now()

        // 可以通过上下文对象,设置一些依附在上下文对象里面的键/值数据
        c.Set("example", "12345")

        // 在这里处理请求到达控制器函数之前的逻辑
     
        // 调用下一个中间件,或者控制器处理函数,具体得看注册了多少个中间件。
        c.Next()

        // 在这里可以处理请求返回给用户之前的逻辑
        latency := time.Since(t)
        log.Print(latency)

        // 例如,查询请求状态吗
        status := c.Writer.Status()
        log.Println(status)
    }
}

func main() {
    r := gin.New()
    // 注册上面自定义的日志中间件
    r.Use(Logger())

    r.GET("/test", func(c *gin.Context) {
        // 查询我们之前在日志中间件,注入的键值数据
        example := c.MustGet("example").(string)

        // it would print: "12345"
        log.Println(example)
    })

    // Listen and serve on 0.0.0.0:8080
    r.Run(":8080")
}

我们可以通过gin的中间件,在请求头设置一个请求id,或者限流等等。


潇尘渊
77 声望19 粉丝

会点php,java,go的菜鸡程序猿