通过上个章节我们已经学会了中间件的用法,在 gin 框架中路由的回调函数、中间件回调函数本质都是一样的,我相信大家使用已经没有任何问题了,那么在本章节我们就由浅入深,继续深度学习中间件.
## 基础用法
### 1.中间件的加载(注册)
注意:Use 函数仅仅只是负责加载(注册)中间件,不负责调用.
```
// 路由可以载入很多个中间件,很多个到底是多少个?
// 想知道答案,那我们继续追踪gin源码去揭晓答案,这里请带着你的疑问向后学习
backend.Use(authorization.CheckTokenAuth(), 下一个中间件,继续下一个中间件, 省略很多个...)
```
## 进阶学习
后续知识点对于初学者来说存在一定的难度,主要是因为在 gin 中,加载的中间件函数: `func ( *gin.Context){ }` 在后续执行时会和注册的路由回调函数在同一个逻辑处执行,中间件函数、路由回调函数都是平行关系,不同的区别是先后顺序不同,gin 对这块逻辑处理是相同的方式,这就需要我们从路由开始追踪,因此涉及到的代码会比较多,过程比较复杂.
学完本章节,gin 最核心的主线逻辑也就彻底搞明白了。
这个过程其实就是对一个 request -> response 的全过程源代码剖析,难度比较大,
### 2.gin 中间件加载过程
```
// Use 的作用就是首先载入中间件回调函数:func(*Context)
// 首先存储起来,然后等着被后续逻辑调用
// 我们使用 goland ,ctrl+鼠标左键,点击 Use 继续追踪gin源码
func (group *RouterGroup) Use(middleware ...HandlerFunc) IRoutes {
group.Handlers = append(group.Handlers, middleware...)
return group.returnObj()
}
// group.Handlers 定义的最终原型如下,本质上就是 func(*Context) 的切片,存储很多个回调函数
type HandlersChain []HandlerFunc // type HandlerFunc func(*Context)
// 学习到这里,你必须知道的就是:
// gin的中间件回调函数全部被存储在 group.Handlers 变量了
// 这里您对该变量有印象就行,后面还需要继续使用该变量
```
习到这里,我们已经知道中间件处理函数被 `Use` 函数注册了在了 `group.Handlers` 变量存储起来了, `Use` 函数可以加载很多个中间件,究竟是多少个,这里我们依然不知道他的具体数量,还有它们什么时候执行,我们也不知道 ... ...
我们只看见 Use 函数最后返回了 `group.returnObj()` ,它是所有路由的处理接口:IRoutes ,已经结束了,我们无法向下追踪了,那就只能从其他地方入手追踪了,既然中间件是加载在具体的路由前面,那么它肯定在某个具体的路由被访问时执行。
### 3.gin 中间件执行逻辑
这个过程很漫长,逻辑比较复杂,我们分步骤分析,去追踪 .
#### 3.1 一个具体的路由地址被请求后的 gin 源码究竟是什么
**3.1.1 定义一个具体的路由以及回调函数**
```
// 1.省略其它无关代码
users.GET("list", func (c *gin.Context){
// 编写该路由路径对应的业务处理回调函数
// 我们省略具体过程 ... ...
})
```
**3.1.2** `**users.GET**` **背后的 gin 源码追踪分析**
```
// 2.在 goland 中,ctrl+鼠标左键 点击 GET 函数,源代码如下:
func (group *RouterGroup) GET(relativePath string, handlers ...HandlerFunc) IRoutes {
// 很明显该函数是 GET + relativePath(开发者定义的路由路径)对应的业务处理逻辑
// 与中间件一样,他的请求回调函数也可以有很多个,具体数量不知道...
// 那么就需要继续追踪 group.handle 源代码
return group.handle(http.MethodGet, relativePath, handlers)
}
// 3.group.handle 函数源码
func (group *RouterGroup) handle(httpMethod, relativePath string, handlers HandlersChain) IRoutes {
// 将定义的路由相对路径拼接为完整的绝对路径
// 因为一个完整的路径往往和前面定义的路由组路径有关系,因此需要一系列拼接
absolutePath := group.calculateAbsolutePath(relativePath)
// 针对完整路由路径将关联的回调函数全部组合出来
// 究竟如何组合,后续继续追踪源码
handlers = group.combineHandlers(handlers)
// 将请求方式(GET、POST等)结合完整路径作为key,处理函数作为 value
// 以 key => value 的形式注册,value 可以是很多个回调函数
group.engine.addRoute(httpMethod, absolutePath, handlers)
return group.returnObj()
}
//4. group.combineHandlers 源代码追踪
func (group *RouterGroup) combineHandlers(handlers HandlersChain) HandlersChain {
// 看到这里还记得 group.Handlers 变量吗?如果不记得请查看本章节 2 标题部分
// finalSize 表示 中间件回调函数的数量 + 具体路由的回调函数数量的总和
finalSize := len(group.Handlers) + len(handlers)
// 如果 finalSize >= abortIndex 就会发生panic,
// abortIndex 的定义值: math.MaxInt8 / 2 ,
// 在go语言中,官方定义: MaxInt8 = 1<<7 - 1, 表示 1*(2^7)-1,最终值:127
// 那么 abortIndex = 127/2 取整 = 63
// 至此我们终于知道, gin 的中间件函数数量 + 路由回调函数的数量总和最大允许 63 个.
if finalSize >= int(abortIndex) {
panic("too many handlers")
}
// 以上条件检查全部通过后,将中间件回调函数和路由回调函数全部合并在一起存储
// HandlersChain 本质就是 [] func(*Context)
mergedHandlers := make(HandlersChain, finalSize)
// group.Handlers 是中间件函数,他在 mergedHandlers 中存储的顺序靠前,也就是索引比较小
copy(mergedHandlers, group.Handlers)
// handlers 是路由回调函数,他的存储位置比中间函数靠后
copy(mergedHandlers[len(group.Handlers):], handlers)
// 最终返回中间件函数+路由函数组合在一起的全部回调函数
return mergedHandlers
}
//5. group.engine.addRoute 源码分析
// gin的路由是一个很复杂的路由前缀树算法模型,完整过程很复杂
// 这里我们主要追踪路由以及回调函数的 存储/注册 过程
func (engine *Engine) addRoute(method, path string, handlers HandlersChain) {
assert1(path[0] == '/', "path must begin with '/'")
assert1(method != "", "HTTP method can not be empty")
assert1(len(handlers) > 0, "there must be at least one handler")
debugPrintRoute(method, path, handlers)
root := engine.trees.get(method)
if root == nil {
root = new(node)
root.fullPath = "/"
engine.trees = append(engine.trees, methodTree{method: method, root: root})
}
// 在这里,gin 将我们定义的完整路径和回调函数进行了注册
// 源码后续继续追、分析
root.addRoute(path, handlers)
// Update maxParams
if paramsCount := countParams(path); paramsCount > engine.maxParams {
engine.maxParams = paramsCount
}
}
//6.root.addRoute(path, handlers) 函数在按照键 => 值进行注册路由与回调函数时调用了很多其他函数
// 这里我将过程函数名字列举如下
func (n *node) addRoute(path string, handlers HandlersChain) {
// 省略其他无关代码
// 又继续调用如下函数
n.insertChild(path, fullPath, handlers)
// ... ...
}
// 省略很多其他的代码
}
// 7. n.insertChild
func (n *node) insertChild(path string, fullPath string, handlers HandlersChain) {
//省略其他无关代码
child := &node{
priority: 1,
fullPath: fullPath, // 完整的路由路径
}
// 最终所有的回调函数
n.handlers = handlers
}
// node 的结构体定义,发现就是自己嵌套自己的一个结构体
// handlers 成员的数据类型(HandlersChain)本质就是 []func(*gin.Context)
// 至此我们彻底明白:路由的存储模型是一个树形结构,每个节点都有自己路由路径以及回调函数 handlers
type node struct {
path string
indices string
wildChild bool
nType nodeType
priority uint32
children []*node // child nodes, at most 1 :param style node at the end of the array
handlers HandlersChain
fullPath string
}
```
- Golang
- Beego框架
- Gin框架
- gin框架介绍
- 使用Gin web框架的知名开源线上项目
- go-admin-gin
- air 热启动
- 完整的form表单参数验证语法
- Go 语言入门练手项目推荐
- Golang是基于多线程模型
- golang 一些概念
- Golang程序开发注意事项
- fatal error: all goroutines are asleep - deadlock
- defer
- Golang 的内建调试器
- go部署
- golang指针重要性
- 包(golang)
- Golang框架选型比较: goframe, beego, iris和gin
- GoFrame
- golang-admin-项目
- go module的使用方法及原理
- go-admin支持多框架的后台系统(go-admin.cn)
- docker gocv
- go-fac
- MSYS2
- 企业开发框架系统推荐
- gorm
- go-zero
- 优秀系统
- GinSkeleton(gin web 及gin 知识)
- 一次 request -> response 的生命周期概述
- 路由与路由组以及gin源码学习
- 中间件以及gin源码学习
- golang项目部署
- 独立部署golang
- 代理部署golang
- 容器部署golang
- golang交叉编译
- goravel
- kardianos+gin 项目作为windows服务运行
- go env
- 适用在Windows、Linux和macOS环境下打包Go应用程序的详细步骤和命令
- Redis
- Dochub
- Docker部署开发go环境
- Docker部署运行go环境
- dochub说明
- Vue
- i18n
- vue3
- vue3基本知识
- element-plus 表格单选
- vue3后台模板
- Thinkphp
- Casbin权限控制中间件
- 容器、依赖注入、门面、事件、中间件
- tp6问答
- 伪静态
- thinkphp-queue
- think-throttle
- thinkphp队列queue的一些使用说明,queue:work和queue:listen的区别
- ThinkPHP6之模型事件的触发条件
- thinkphp-swoole
- save、update、insert 的区别
- Socket
- workerman
- 介绍
- 从ThinkPHP6移植到Webman的一些技术和经验(干货)
- swoole
- swoole介绍
- hyperf
- hf官网
- Swoft
- swoft官网
- easyswoole
- easyswoole官网地址
- EASYSWOOLE 聊天室DEMO
- socket问答
- MySQL
- 聚簇索引与非聚簇索引
- Mysql使用max获取最大值细节
- 主从复制
- 随机生成20万User表的数据
- MySQL进阶-----前缀索引、单例与联合索引
- PHP
- 面向切面编程AOP
- php是单线程的一定程度上也可以看成是“多线程”
- PHP 线程,进程、并发、并行 的理解
- excel数据画表格图片
- php第三方包
- monolog/monolog
- league/glide
- 博客-知识网站
- php 常用bc函数
- PHP知识点的应用场景
- AOP(面向切面编程)
- 注解
- 依赖注入
- 事件机制
- phpspreadsheet导出数据和图片到excel
- Hyperf
- mineAdmin
- 微服务
- nacos注册服务
- simps-mqtt连接客户端simps
- Linux
- 切换php版本
- Vim
- Laravel
- RabbitMQ
- thinkphp+rabbitmq
- 博客
- Webman框架
- 框架注意问题
- 关于内存泄漏
- 移动端自动化
- 懒人精灵
- 工具应用
- render
- gitlab Sourcetree
- ssh-agent失败 错误代码-1
- 资源网站
- Git
- wkhtmltopdf
- MSYS2 介绍
- powershell curl 使用教程
- NSSM(windows服务工具)
- MinGW64
- 知识扩展
- 对象存储系统
- minio
- 雪花ID
- 请求body参数类型
- GraphQL
- js 深拷贝
- window 共享 centos文件夹
- 前端get/post 请求 特殊符号 “+”传参数问题
- 什么是SCM系统?SCM系统与ERP系统有什么区别?
- nginx 日志格式统一为 json
- 特殊符号怎么打
- 收藏网址
- 收藏-golang
- 收藏-vue3
- 收藏-php
- 收藏-node
- 收藏-前端
- 规划ITEM
- 旅游类
- 人脸识别
- dlib
- Docker&&部署
- Docker-compose
- Docker的网络模式
- rancher
- DHorse
- Elasticsearch
- es与kibana都docke连接
- 4种数据同步到Elasticsearch方案
- GPT
- 推荐系统
- fastposter海报生成
- elasticsearch+logstash+kibana
- beego文档系统-MinDoc
- jeecg开源平台
- Java
- 打包部署
- spring boot
- 依赖
- Maven 相关 命令
- Gradle 相关命令
- mybatis
- mybatis.plus
- spring boot 模板引擎
- SpringBoot+Maven多模块项目(创建、依赖、打包可执行jar包部署测试)完整流程
- Spring Cloud
- Sentinel
- nacos
- Apollo
- java推荐项目
- gradle
- Maven
- Nexus仓库管理器
- Python
- Masonite框架
- scrapy
- Python2的pip2
- Python3 安装 pip3
- 安全攻防
- 运维技术
- 腾讯云安全加固建议
- 免费freessl证书申请
- ruby
- homeland
- Protobuf
- GIT
- FFMPEG
- 命令说明
- 音频
- ffmpeg合并多个MP4视频
- NODEJS
- 开发npm包
- MongoDB
- php-docker-mongodb环境搭建
- mongo基本命令
- Docker安装MongoDB最新版并连接
- 少儿编程官网
- UI推荐
- MQTT
- PHP连接mqtt
- EMQX服务端
- php搭建mqtt服务端