手把手和你一起实现一个Web框架实战——EzWeb框架(五)[Go语言笔记]Go项目实战
阅读原文时间:2021年08月21日阅读:2

手把手和你一起实现一个Web框架实战——EzWeb框架(五)[Go语言笔记]Go项目实战

代码仓库:

github

gitee

中文注释,非常详尽,可以配合食用

本篇代码,请选择demo5

中间件实现

一、Context设计

type Context struct {
   Writer http.ResponseWriter
   Req *http.Request
   //请求的信息,包括路由和方法
   Path string
   Method string
   Params map[string]string   /*用于存储外面拿到的参数 ":xxx" or "*xxx" */
   //响应的状态码
   StatusCode int
   //中间件
   handlers []HandlerFunc
   index    int   /* 用于记录当前执行到第几个中间件 */
}

我们每次请求生成的context,我们选择在其中放入和我们中间件和执行控制变量index

二、中间件对路由组的注册方法

// 将路中间件,放入路由组的中间件方法切片中
func (group *RouterGroup) Use(middlewares ...HandlerFunc)  {
    group.middlewares = append(group.middlewares, middlewares...)
}

当每个请求到来后,ServeHTTP函数执行时,它将该生成一个context并将进行传入的url和路由组前缀做前缀对比,找到满足条件的路由组,取出它的中间件,然后存入到生成的context。

// ServeHTTP 方法的实现,用于实现处理HTTP请求
func (engine *Engine) ServeHTTP(w http.ResponseWriter, req *http.Request) {
    var middlewares []HandlerFunc
    for _, group := range engine.groups {
        //比对路由组存的前缀和请求路径,把属于这个请求映射的路由组中的中间件取到
        //意思就是比对发现该请求属于哪一个路由组,需要哪些中间件,取出来执行
        if strings.HasPrefix(req.URL.Path, group.prefix) {
            middlewares = append(middlewares, group.middlewares...)
        }
    }
    //根据req和w实例一个context
    c := newContext(w, req)
    //将取道的中间件赋给这个context
    c.handlers = middlewares
    //通过封装好的context执行处理
    engine.router.handle(c)
}

三、处理函数

在处理函数handle()中,我们根据路由拿到已经注册的方法,放入到中间件后,在通过Next函数进行处理

//根据context中存储的 c.Method 和 c.Path 拿到对应的处理方法,进行执行,如果拿到的路由没有注册,则返回404
func (r *router) handle(c *Context)  {
    //获取匹配到的节点,同时也拿到两类动态路由中参数
    n, params := r.getRoute(c.Method, c.Path)
    if n != nil {
        c.Params = params
        //拿目的节点中的path做key来找handlers
        key := c.Method + "-" + n.path
        //根据路径拿到处理器
        c.handlers = append(c.handlers, r.handlers[key])
    }else {
        //不存在节点的情况下,给生成的c加入一个404方法
        c.handlers = append(c.handlers, func(c *Context) {
            c.String(http.StatusNotFound, "404 NOT FOUND: ", c.Path)
        })
    }
    c.Next()
}

四、Next方法执行处理

index进行控制,遍历完c.handlers中存储的方法执行。

// 当中间件调用了 Next 方法时,就往后执行下一个,同时index交由下一个中间件控制
func (c *Context) Next()  {
    c.index++
    //执行完之后所有的handlers
    for ;c.index < len(c.handlers); c.index++{
        c.handlers[c.index](c)
    }
}

这里的日志中间件在Next中被执行时,也调用了Next函数。这里第四行的执行巧妙的做到了,控制index++,执行下一个中间件,直到执行完了又回到Logger执行,回到原来的Next的for循环,发现不满足继续循环的条件,然后退出。

func Logger() HandlerFunc {
   return func(c *Context) {
      t := time.Now()
      c.Next()
      log.Printf("[%d] %s in %v", c.StatusCode, c.Req.RequestURI, time.Since(t))
   }
}

demo测试:

/*
@Time : 2021/8/16 下午4:01
@Author : mrxuexi
@File : main
@Software: GoLand
*/
package main

import (
    "Ez"
    "net/http"
)
func main() {
    r := Ez.New()
    //给所有的路由组都添加了中间件logger
    r.Use(Ez.Logger())
    api := r.Group("/api")

    api.POST("/hello", func(c *Ez.Context) {
        c.JSON(http.StatusOK,Ez.H{
                "message" : "hello",
        })
    })
//next的应用
    api.Use(func(c *Ez.Context) {
        c.JSON(200,Ez.H{
            "test" : "middleware2-1",
        })
        c.Next()
        c.JSON(200, Ez.H{
            "test" : "middleware2-2",
        })
    })

    r.Run(":9090")
}

成功!

参考:

[1]: https://github.com/geektutu/7days-golang/tree/master/gee-web ""gee""

[2]: https://github.com/gin-gonic/gin ""gin""