gin框架路由拆分与注册
转自
gin框架路由拆分与注册
基本的路由注册
下面最基础的gin路由注册方式,适用于路由条目比较少的简单项目或者项目demo。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | package main import ( "net/http" "github.com/gin-gonic/gin" ) func helloHandler(c *gin.Context) { c.JSON(http.StatusOK, gin.H{ "message" : "Hello q1mi!" , }) } func main() { r := gin.Default() r.GET( "/hello" , helloHandler) if err := r.Run(); err != nil { fmt.Println( "startup service failed, err:%v\n" , err) } } |
路由拆分成单独文件或包
当项目的规模增大后就不太适合继续在项目的main.go
文件中去实现路由注册相关逻辑了,我们会倾向于把路由部分的代码都拆分出来,形成一个单独的文件或包:
我们在routers.go
文件中定义并注册路由信息:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | package main import ( "net/http" "github.com/gin-gonic/gin" ) func helloHandler(c *gin.Context) { c.JSON(http.StatusOK, gin.H{ "message" : "Hello q1mi!" , }) } func setupRouter() *gin.Engine { r := gin.Default() r.GET( "/hello" , helloHandler) return r } |
此时main.go
中调用上面定义好的setupRouter
函数:
1 2 3 4 5 6 | func main() { r := setupRouter() if err := r.Run(); err != nil { fmt.Println( "startup service failed, err:%v\n" , err) } } |
此时的目录结构:
gin_demo
├── go.mod
├── go.sum
├── main.go
└── routers.go
把路由部分的代码单独拆分成包的话也是可以的,拆分后的目录结构如下:
gin_demo
├── go.mod
├── go.sum
├── main.go
└── routers
└── routers.go
routers/routers.go
需要注意此时setupRouter
需要改成首字母大写:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | package routers import ( "net/http" "github.com/gin-gonic/gin" ) func helloHandler(c *gin.Context) { c.JSON(http.StatusOK, gin.H{ "message" : "Hello q1mi!" , }) } // SetupRouter 配置路由信息 func SetupRouter() *gin.Engine { r := gin.Default() r.GET( "/hello" , helloHandler) return r } |
main.go
文件内容如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 | package main import ( "fmt" "gin_demo/routers" ) func main() { r := routers.SetupRouter() if err := r.Run(); err != nil { fmt.Println( "startup service failed, err:%v\n" , err) } } |
路由拆分成多个文件
当我们的业务规模继续膨胀,单独的一个routers
文件或包已经满足不了我们的需求了,
1 2 3 4 5 6 7 8 | func SetupRouter() *gin.Engine { r := gin.Default() r.GET( "/hello" , helloHandler) r.GET( "/xx1" , xxHandler1) ... r.GET( "/xx30" , xxHandler30) return r } |
因为我们把所有的路由注册都写在一个SetupRouter
函数中的话就会太复杂了。
我们可以分开定义多个路由文件,例如:
gin_demo
├── go.mod
├── go.sum
├── main.go
└── routers
├── blog.go
└── shop.go
routers/shop.go
中添加一个LoadShop
的函数,将shop相关的路由注册到指定的路由器:
1 2 3 4 5 6 | func LoadShop(e *gin.Engine) { e.GET( "/hello" , helloHandler) e.GET( "/goods" , goodsHandler) e.GET( "/checkout" , checkoutHandler) ... } |
routers/blog.go
中添加一个`LoadBlog的函数,将blog相关的路由注册到指定的路由器:
1 2 3 4 5 | func LoadBlog(e *gin.Engine) { e.GET( "/post" , postHandler) e.GET( "/comment" , commentHandler) ... } |
在main函数中实现最终的注册逻辑如下:
1 2 3 4 5 6 7 8 | func main() { r := gin.Default() routers.LoadBlog(r) routers.LoadShop(r) if err := r.Run(); err != nil { fmt.Println( "startup service failed, err:%v\n" , err) } } |
路由拆分到不同的APP
有时候项目规模实在太大,那么我们就更倾向于把业务拆分的更详细一些,例如把不同的业务代码拆分成不同的APP。
因此我们在项目目录下单独定义一个app
目录,用来存放我们不同业务线的代码文件,这样就很容易进行横向扩展。大致目录结构如下:
gin_demo
├── app
│ ├── blog
│ │ ├── handler.go
│ │ └── router.go
│ └── shop
│ ├── handler.go
│ └── router.go
├── go.mod
├── go.sum
├── main.go
└── routers
└── routers.go
其中app/blog/router.go
用来定义post相关路由信息,具体内容如下:
1 2 3 4 | func Routers(e *gin.Engine) { e.GET( "/post" , postHandler) e.GET( "/comment" , commentHandler) } |
app/shop/router.go
用来定义shop相关路由信息,具体内容如下:
1 2 3 4 | func Routers(e *gin.Engine) { e.GET( "/goods" , goodsHandler) e.GET( "/checkout" , checkoutHandler) } |
routers/routers.go
中根据需要定义Include
函数用来注册子app中定义的路由,Init
函数用来进行路由的初始化操作:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | type Option func (*gin.Engine) var options = []Option{} // 注册app的路由配置 func Include(opts ...Option) { options = append(options, opts...) } // 初始化 func Init() *gin.Engine { r := gin.New() for _, opt := range options { opt(r) } return r } |
main.go
中按如下方式先注册子app中的路由,然后再进行路由的初始化:
1 2 3 4 5 6 7 8 9 | func main() { // 加载多个APP的路由配置 routers.Include(shop.Routers, blog.Routers) // 初始化路由 r := routers.Init() if err := r.Run(); err != nil { fmt.Println( "startup service failed, err:%v\n" , err) } } |
总结
gin
框架是一个非常容易扩展的web框架,本文是我在日常编码中总结的一点点经验,因为世界上不可能有完全相同的项目,每个人也都有自己的编程习惯,关于gin框架路由注册的方式我就在此抛砖引玉了。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 全程不用写代码,我用AI程序员写了一个飞机大战
· DeepSeek 开源周回顾「GitHub 热点速览」
· 记一次.NET内存居高不下排查解决与启示
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· .NET10 - 预览版1新功能体验(一)