手把手和你一起实现一个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""