-
手把手和你一起实现一个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""
__EOF__
栏目列表
最新更新
nodejs爬虫
Python正则表达式完全指南
爬取豆瓣Top250图书数据
shp 地图文件批量添加字段
爬虫小试牛刀(爬取学校通知公告)
【python基础】函数-初识函数
【python基础】函数-返回值
HTTP请求:requests模块基础使用必知必会
Python初学者友好丨详解参数传递类型
如何有效管理爬虫流量?
SQL SERVER中递归
2个场景实例讲解GaussDB(DWS)基表统计信息估
常用的 SQL Server 关键字及其含义
动手分析SQL Server中的事务中使用的锁
openGauss内核分析:SQL by pass & 经典执行
一招教你如何高效批量导入与更新数据
天天写SQL,这些神奇的特性你知道吗?
openGauss内核分析:执行计划生成
[IM002]Navicat ODBC驱动器管理器 未发现数据
初入Sql Server 之 存储过程的简单使用
这是目前我见过最好的跨域解决方案!
减少回流与重绘
减少回流与重绘
如何使用KrpanoToolJS在浏览器切图
performance.now() 与 Date.now() 对比
一款纯 JS 实现的轻量化图片编辑器
关于开发 VS Code 插件遇到的 workbench.scm.
前端设计模式——观察者模式
前端设计模式——中介者模式
创建型-原型模式