Gin 中间件

中间件 middleware 在 Golang 中是一个很重要的概念,与 Java 中的拦截器类似,常用于提高应用程序的扩展能力,留出更多的扩展空间,比如:日志记录、故障处理等功能。

在 Gin 的整个实现中,中间件是 Gin 的精髓。一个个中间件组成了一条中间件链,对 HTTP Request 请求进行拦截处理,实现了代码的解耦和分离,并且中间件之间相互无感知,每个中间件只需要处理自己需要处理的事情即可。

Gin 中间件概述

在最初的示例中,我们是直接通过 gin.Default() 来初始化 gin 对象,其中它包含了一个自带默认中间件的 *Engine

其中,Default() 函数会默认绑定两个已经准备好的中间件,它们就是 Logger 和 Recovery,帮助我们打印日志输出和 painc 处理。

// Default returns an Engine instance with the Logger and Recovery middleware already attached.
func Default() *Engine {
    debugPrintWARNINGDefault()
    engine := New()
    engine.Use(Logger(), Recovery())
    return engine
}

从上面 Default()函数中,我们可以看到 Gin 中间件是通过 Use 方法设置的,它是一个可变参数,可同时设置多个中间件。

// 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 定义的一个函数 func,该 func 的返回值类型为 HandlerFunc

中间件实现

中间件主要用于完成一些通用的功能,便于功能的扩充、统一实现,类似于 filter。

从上面的介绍中,要实现一个中间件,只需满足以下两点:

  • 它是一个函数
  • 函数返回值的类型必须是 HandlerFunc

示例

实现计算每次请求执行花费的时间。

package main

import (
    "fmt"
    "time"

    "github.com/gin-gonic/gin"
)

func main() {
    // 初始化gin对象
    route := gin.Default()

    // 中间件注册
    route.Use(costTimeMiddleware())

    // 设置一个get请求,其URL为/hello,并实现简单的响应
    route.GET("/hello", func(c *gin.Context) {
        c.JSON(200, gin.H{
            "message": "hello world!",
        })
    })

    // 启动服务
    route.Run()
}

func costTimeMiddleware() gin.HandlerFunc {
    return func(c *gin.Context) {
        // 请求前获取当前时间
        nowTime := time.Now()

        // 请求处理
        c.Next()

        // 请求处理完获取花费的时间
        costTime := time.Since(nowTime)

        requestURL := c.Request.URL.String()
        fmt.Printf("the request URL %s cost %v\n", requestURL, costTime)
    }
}

其中,c.Next() 只允许在中间件函数中使用,用于挂起请求业务逻辑处理。为了更好的理解中间件函数的执行,可以将中间件函数分为三层:

gin中间件函数执行流程
图 9.2.3.1:gin中间件函数执行流程

根据中间件注册的位置不同,分为:

  • 全局中间件:所有请求都经过该中间件。例如,上述示例中就属于全局中间件。
  • 局部中间件:只针对注册的路由生效。如果将中间件 costTimeMiddleware 注册在如下位置,则属于局部中间件,只对请求 /hello 生效。

    route.GET("/hello", costTimeMiddleware(), func(c *gin.Context) {
        c.JSON(200, gin.H{
            "message": "hello world!",
        })
    })
    

常见中间件

可参考:contrib,里面包含众多常用的中间件。

Copyright © xcbeyond.cn 2021 all right reserved,powered by Gitbook Updated at 2021-08-10 23:59:29

results matching ""

    No results matching ""