如何构建一个有效的中间件
组成部分
中间件通常由两部分组成:
-
第一部分在初始化中间件时执行一次。在这里设置全局对象、配置逻辑等——所有只需要在应用生命周期中发生一次的事情。
-
第二部分在每个请求时执行。例如,在数据库中间件中,你会将全局数据库对象注入到请求上下文中。一旦它在上下文中,其他中间件和处理函数就可以检索和使用它。
func funcName(params string) gin.HandlerFunc { // <--- // This is part one // ---> // Example initialization: validate input params if err := check(params); err != nil { panic(err) }
return func(c *gin.Context) { // <--- // This is part two // ---> // Example execution per request: inject into context c.Set("TestVar", params) c.Next() }}执行过程
让我们看下面的示例代码:
func main() { router := gin.Default()
router.Use(globalMiddleware())
router.GET("/rest/n/api/*some", mid1(), mid2(), handler)
router.Run()}
func globalMiddleware() gin.HandlerFunc { fmt.Println("globalMiddleware...1")
return func(c *gin.Context) { fmt.Println("globalMiddleware...2") c.Next() fmt.Println("globalMiddleware...3") }}
func handler(c *gin.Context) { fmt.Println("exec handler.")}
func mid1() gin.HandlerFunc { fmt.Println("mid1...1")
return func(c *gin.Context) {
fmt.Println("mid1...2") c.Next() fmt.Println("mid1...3") }}
func mid2() gin.HandlerFunc { fmt.Println("mid2...1")
return func(c *gin.Context) { fmt.Println("mid2...2") c.Next() fmt.Println("mid2...3") }}根据上面组成部分中的说明,当你运行 Gin 进程时,每个中间件的第一部分首先执行,并打印以下信息:
globalMiddleware...1mid1...1mid2...1初始化顺序为:
globalMiddleware...1 | vmid1...1 | vmid2...1当你发起请求时——例如 curl -v localhost:8080/rest/n/api/some——每个中间件的第二部分按顺序执行,并输出以下内容:
globalMiddleware...2mid1...2mid2...2exec handler.mid2...3mid1...3globalMiddleware...3换句话说,执行顺序为:
globalMiddleware...2 | vmid1...2 | vmid2...2 | vexec handler. | vmid2...3 | vmid1...3 | vglobalMiddleware...3