Go Web 框架 Gin 路由的学习(go语言 web框架)
ccwgpt 2024-09-18 12:19 34 浏览 0 评论
Gin 是目前应用比较广泛的Golang web 框架。目前,Github Star 数已经达到了3.8w. 框架的实现非常简单,可定制性非常强,性能也比较好,深受golang开发者的喜爱。Gin 提供了web开发的一些基本功能。如路由,中间件,日志,参数获取等,本文主要从源码的角度分析Gin的路由实现。
Gin 的路由功能是基于 https://github.com/julienschmidt/httprouter 这个项目实现的。目前也
有很多其他Web框架也基于该路由框架做了二次开发。
http 路由的接口
在 Gin 中,为了兼容不同路由的引擎,定义了 IRoutes 和 IRouter 接口,便于替换其他的路由实现。(目前默认是httprouter)
下面是一个路由的接口定义:
type IRoutes interface {
Use(...HandlerFunc) IRoutes
Handle(string, string, ...HandlerFunc) IRoutes
Any(string, ...HandlerFunc) IRoutes
GET(string, ...HandlerFunc) IRoutes
POST(string, ...HandlerFunc) IRoutes
DELETE(string, ...HandlerFunc) IRoutes
PATCH(string, ...HandlerFunc) IRoutes
PUT(string, ...HandlerFunc) IRoutes
OPTIONS(string, ...HandlerFunc) IRoutes
HEAD(string, ...HandlerFunc) IRoutes
StaticFile(string, string) IRoutes
Static(string, string) IRoutes
StaticFS(string, http.FileSystem) IRoutes
}
type HandlerFunc func(*Context)
HandlerFunc 是一个方法类型的定义,我们定义的路由其实就是一个路径与HandlerFunc 的映射关系。从上面的定义可以看出,IRoutes 主要定义了一些基于http方法、静态方法的路径和一组方法的映射。Use 方法是针对此路由的所有路径映射一组方法,在使用上是为了给这些路由添加中间件。
除了上面的定义外,Gin 还有路由组的抽象。
type IRouter interface {
IRoutes
Group(string, ...HandlerFunc) *RouterGroup
}
路由组是在IRoutes 的基础上,有了组的概念,组下面还可以挂在不同的组。组的概念可以很好的管理一组路由,路由组可以自己定义一套Handler方法(即一组中间件)。
个人认为IRouter的定义Group 应该返回 IRouter,这样可以把路由组更加抽象,也不会改变现有服务的使用。期待看下Gin源码什么时候会按照这种定义方法修改过来。
在Gin框架中,路由由 RouterGroup 实现。我们从构造和路由查找两个方面分析路由的实现。
路由实现
路由的本质就是在给定 路径与Handler映射关系 的前提下,当提供新的url时,给出对应func 的过程。其中可能需要从url中提取参数,或者按照 * 匹配 url 的情况。
首先,我们看下Gin中路由结构的定义。
// gin engine
type Engine struct {
RouterGroup
// ... 其他字段
trees methodTrees
}
// 每个 http 方法定义一个森林
type methodTrees []methodTree
type methodTree struct {
method string
root *node
}
// 路由组的定义
type RouterGroup struct {
Handlers HandlersChain
basePath string
engine *Engine
root bool
}
从定义中可以看出,其实Gin 的 Engine 是复用了 RouterGroup。对于不同的 http method,都通过一个森林来存储路由数据。下面是森林上每个节点的定义:
type node struct {
path string // 当前路径
indices string // 对应children 的前缀
wildChild bool // 可能是带参数的,或者是 * 的,所以是野节点
nType nodeType // 参数节点,静态节点
priority uint32 // 优先级 ,优先级高的放在children 放在前面。
children []*node // 子节点
handlers HandlersChain // 调用链
fullPath string // 全路径
}
从代码实现上得知,这个森林其实是一个压缩版本的Trie树,每个节点会存储前缀相同的路径数据。下面,我们通过代码来学习下路由的添加和删除。
路由的添加
路由的添加,就是将path路径添加到定义的Trie树种,将handlers 添加到对应的node 节点。
func (n *node) addRoute(path string, handlers HandlersChain) {
// 初始化和维护优先级
for {
// 查找前缀
i := longestCommonPrefix(path, n.path)
// 原有路径长的情况下
// 节点n 的 path 变为了公共前缀
// 原有n 的path 路径变为了现有n 的子节点
// 当添加的path长的情况
// 需要分情况讨论:
// 1. 如果是一个带参数的路径,校验是否后续路径不同,如果不同则继续扫描下一段路径
// 2. 如果是带 * 的路径, 则直接报错
// 3. 如果已经有对应的首字母,修改当前node节点,并继续扫描,并扫描下一段路径
// 4. 如果非参数或者 * 匹配的方法,则插入一个子节点路径,并完成扫描
// 最后注册handlers,添加fullPath
n.handlers = handlers
n.fullPath = fullPath
return
}
}
从上面的代码注释可以看出,路由的添加,主要是通过不断对比当前节点的path和添加的path,做添加节点或者节点变更的操作,达到添加path的目的。
路径查找
在服务请求时,路由的责任就是给定一个url请求,拿到节点保存的handlers,以及url中包含的参数值。下面是对一个url 的解析实现。
type nodeValue struct {
handlers HandlersChain
params *Params
tsr bool
fullPath string
}
func (n *node) getValue(path string, params *Params, unescape bool) (value nodeValue) {
walk: // Outer loop for walking the tree
for {
prefix := n.path
// 如果比当前节点路径要长:
// - 非参数类型或模糊匹配的URL,如果和当前节点前缀匹配,直接查看 node 的子节点
// - 参数化的node, 按照 / 分割提取参数,如果未结束,则继续匹配剩下的路径,否则返回结果。
// - * 匹配的node,将剩余的路径添加到 param 中直接返回。
// 如果和当前节点相等,那就直接返回即可。
// 这里还做了非本方法的路径匹配,用户返回http 方法错误的异常报告。
}
}
一个例子
下面通过一个例子,方便我们快速理解router的实现。
加入下面的一个路径:
/search/
/support/
/blog/:post/
/about-us/team/
/contact/
在树中,我们看到的样子如下:
Path
\
├s
|├earch\
|└upport\
├blog\
| └:post
| └\
├about-us\
| └team\
└contact\
在做路由查找时,通过路径不断匹配,找到对应的子节点。拿到对应子节点下的handler。完成路由的匹配。
总结
- httprouter 没有实现了routergroup功能,只是实现了router 的功能,在gin中做了实现
- 通过Trie树实现路由是比较基础的一种实现方法,除了这种方法外,还可以考虑通过正则的方式提取路由。
- Gin http 服务是基于 Go 的 net/http 库的, net/http 库中handler 的实现是针对不同的 http method 的,所以需要在engine 中针对不同的method 提供不同的trie 树。
- 在添加路由时,如果使用了 any 方法,则在每个http method 下都会添加一样的路径。
- middleware 本质上只是一个 HandlerFunc.
相关推荐
- 一个基于.Net Core遵循Clean Architecture原则开源架构
-
今天给大家推荐一个遵循CleanArchitecture原则开源架构。项目简介这是基于Asp.netCore6开发的,遵循CleanArchitecture原则,可以高效、快速地构建基于Ra...
- AI写代码翻车无数次,我发现只要提前做好这3步,bug立减80%
-
写十万行全是bug之后终于找到方法了开发"提示词管理助手"新版本那会儿,我差点被bug整崩溃。刚开始两周,全靠AI改代码架构,结果十万行程序漏洞百出。本来以为AI说没问题就稳了,结果...
- OneCode低代码平台的事件驱动设计:架构解析与实践
-
引言:低代码平台的事件驱动范式在现代软件开发中,事件驱动架构(EDA)已成为构建灵活、松耦合系统的核心范式。OneCode低代码平台通过创新性的注解驱动设计,将事件驱动理念深度融入平台架构,实现了业务...
- 国内大厂AI插件评测:根据UI图生成Vue前端代码
-
在IDEA中安装大厂的AI插件,打开ruoyi增强项目:yudao-ui-admin-vue31.CodeBuddy插件登录腾讯的CodeBuddy后,大模型选择deepseek-v3,输入提示语:...
- AI+低代码技术揭秘(二):核心架构
-
本文档介绍了为VTJ低代码平台提供支持的基本架构组件,包括Engine编排层、Provider服务系统、数据模型和代码生成管道。有关UI组件库和widget系统的信息,请参阅UI...
- GitDiagram用AI把代码库变成可视化架构图
-
这是一个名为gitdiagram的开源工具,可将GitHub仓库实时转换为交互式架构图,帮助开发者快速理解代码结构。核心功能一键可视化:替换GitHubURL中的"hub...
- 30天自制操作系统:第六天:代码架构整理与中断处理
-
1.拆开bootpack.c文件。根据设计模式将对应的功能封装成独立的文件。2.初始化pic:pic(可编程中断控制器):在设计上,cpu单独只能处理一个中断。而pic是将8个中断信号集合成一个中断...
- AI写代码越帮越忙?2025年研究揭露惊人真相
-
近年来,AI工具如雨后春笋般涌现,许多人开始幻想程序员的未来就是“对着AI说几句话”,就能轻松写出完美的代码。然而,2025年的一项最新研究却颠覆了这一期待,揭示了一个令人意外的结果。研究邀请了16位...
- 一键理解开源项目:两个自动生成GitHub代码架构图与说明书工具
-
一、GitDiagram可以一键生成github代码仓库的架构图如果想要可视化github开源项目:https://github.com/luler/reflex_ai_fast,也可以直接把域名替换...
- 5分钟掌握 c# 网络通讯架构及代码示例
-
以下是C#网络通讯架构的核心要点及代码示例,按协议类型分类整理:一、TCP协议(可靠连接)1.同步通信//服务器端usingSystem.Net.Sockets;usingTcpListene...
- 从复杂到优雅:用建造者和责任链重塑代码架构
-
引用设计模式是软件开发中的重要工具,它为解决常见问题提供了标准化的解决方案,提高了代码的可维护性和可扩展性,提升了开发效率,促进了团队协作,提高了软件质量,并帮助开发者更好地适应需求变化。通过学习和应...
- 低代码开发当道,我还需要学习LangChain这些框架吗?| IT杂谈
-
专注LLM深度应用,关注我不迷路前两天有位兄弟问了个问题:当然我很能理解这位朋友的担忧:期望效率最大化,时间用在刀刃上,“不要重新发明轮子”嘛。铺天盖地的AI信息轰炸与概念炒作,很容易让人浮躁与迷茫。...
- 框架设计并不是简单粗暴地写代码,而是要先弄清逻辑
-
3.框架设计3.框架设计本节我们要开发一个UI框架,底层以白鹭引擎为例。框架设计的第一步并不是直接撸代码,而是先想清楚设计思想,抽象。一个一个的UI窗口是独立的吗?不是的,...
- 大佬用 Avalonia 框架开发的 C# 代码 IDE
-
AvalonStudioAvalonStudio是一个开源的跨平台的开发编辑器(IDE),AvalonStudio的目标是成为一个功能齐全,并且可以让开发者快速使用的IDE,提高开发的生产力。A...
- 轻量级框架Lagent 仅需20行代码即可构建自己的智能代理
-
站长之家(ChinaZ.com)8月30日消息:Lagent是一个专注于基于LLM模型的代理开发的轻量级框架。它的设计旨在简化和提高这种模型下代理的开发效率。LLM模型是一种强大的工具,可以...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- 框架图 (58)
- flask框架 (53)
- quartz框架 (51)
- abp框架 (47)
- springmvc框架 (49)
- 分布式事务框架 (65)
- scrapy框架 (56)
- shiro框架 (61)
- 定时任务框架 (56)
- java日志框架 (61)
- mfc框架 (52)
- abb框架断路器 (48)
- beego框架 (52)
- java框架spring (58)
- grpc框架 (65)
- tornado框架 (48)
- 前端框架bootstrap (54)
- orm框架有哪些 (51)
- 知识框架图 (52)
- ppt框架 (55)
- 框架图模板 (59)
- 内联框架 (52)
- cad怎么画框架 (58)
- ssm框架实现登录注册 (49)
- oracle字符串长度 (48)