1. 路由拆分与注册

1.1.1. 基本的路由注册

下面最基础的gin路由注册方式,适用于路由条目比较少的简单项目或者项目demo。

  1. package main
  2. import (
  3. "net/http"
  4. "github.com/gin-gonic/gin"
  5. )
  6. func helloHandler(c *gin.Context) {
  7. c.JSON(http.StatusOK, gin.H{
  8. "message": "Hello www.topgoer.com!",
  9. })
  10. }
  11. func main() {
  12. r := gin.Default()
  13. r.GET("/topgoer", helloHandler)
  14. if err := r.Run(); err != nil {
  15. fmt.Println("startup service failed, err:%v\n", err)
  16. }
  17. }

1.1.2. 路由拆分成单独文件或包

当项目的规模增大后就不太适合继续在项目的main.go文件中去实现路由注册相关逻辑了,我们会倾向于把路由部分的代码都拆分出来,形成一个单独的文件或包:

我们在routers.go文件中定义并注册路由信息:

  1. package main
  2. import (
  3. "net/http"
  4. "github.com/gin-gonic/gin"
  5. )
  6. func helloHandler(c *gin.Context) {
  7. c.JSON(http.StatusOK, gin.H{
  8. "message": "Hello www.topgoer.com!",
  9. })
  10. }
  11. func setupRouter() *gin.Engine {
  12. r := gin.Default()
  13. r.GET("/topgoer", helloHandler)
  14. return r
  15. }

此时main.go中调用上面定义好的setupRouter函数:

  1. func main() {
  2. r := setupRouter()
  3. if err := r.Run(); err != nil {
  4. fmt.Println("startup service failed, err:%v\n", err)
  5. }
  6. }

此时的目录结构:

  1. gin_demo
  2. ├── go.mod
  3. ├── go.sum
  4. ├── main.go
  5. └── routers.go

把路由部分的代码单独拆分成包的话也是可以的,拆分后的目录结构如下:

  1. gin_demo
  2. ├── go.mod
  3. ├── go.sum
  4. ├── main.go
  5. └── routers
  6. └── routers.go

routers/routers.go需要注意此时setupRouter需要改成首字母大写:

  1. package routers
  2. import (
  3. "net/http"
  4. "github.com/gin-gonic/gin"
  5. )
  6. func helloHandler(c *gin.Context) {
  7. c.JSON(http.StatusOK, gin.H{
  8. "message": "Hello www.topgoer.com",
  9. })
  10. }
  11. // SetupRouter 配置路由信息
  12. func SetupRouter() *gin.Engine {
  13. r := gin.Default()
  14. r.GET("/topgoer", helloHandler)
  15. return r
  16. }

main.go文件内容如下:

  1. package main
  2. import (
  3. "fmt"
  4. "gin_demo/routers"
  5. )
  6. func main() {
  7. r := routers.SetupRouter()
  8. if err := r.Run(); err != nil {
  9. fmt.Println("startup service failed, err:%v\n", err)
  10. }
  11. }

1.1.3. 路由拆分成多个文件

当我们的业务规模继续膨胀,单独的一个routers文件或包已经满足不了我们的需求了,

  1. func SetupRouter() *gin.Engine {
  2. r := gin.Default()
  3. r.GET("/topgoer", helloHandler)
  4. r.GET("/xx1", xxHandler1)
  5. ...
  6. r.GET("/xx30", xxHandler30)
  7. return r
  8. }

因为我们把所有的路由注册都写在一个SetupRouter函数中的话就会太复杂了。

我们可以分开定义多个路由文件,例如:

  1. gin_demo
  2. ├── go.mod
  3. ├── go.sum
  4. ├── main.go
  5. └── routers
  6. ├── blog.go
  7. └── shop.go

routers/shop.go中添加一个LoadShop的函数,将shop相关的路由注册到指定的路由器:

  1. func LoadShop(e *gin.Engine) {
  2. e.GET("/hello", helloHandler)
  3. e.GET("/goods", goodsHandler)
  4. e.GET("/checkout", checkoutHandler)
  5. ...
  6. }

routers/blog.go中添加一个LoadBlog的函数,将blog相关的路由注册到指定的路由器:

  1. func LoadBlog(e *gin.Engine) {
  2. e.GET("/post", postHandler)
  3. e.GET("/comment", commentHandler)
  4. ...
  5. }

在main函数中实现最终的注册逻辑如下:

  1. func main() {
  2. r := gin.Default()
  3. routers.LoadBlog(r)
  4. routers.LoadShop(r)
  5. if err := r.Run(); err != nil {
  6. fmt.Println("startup service failed, err:%v\n", err)
  7. }
  8. }

1.1.4. 路由拆分到不同的APP

有时候项目规模实在太大,那么我们就更倾向于把业务拆分的更详细一些,例如把不同的业务代码拆分成不同的APP。

因此我们在项目目录下单独定义一个app目录,用来存放我们不同业务线的代码文件,这样就很容易进行横向扩展。大致目录结构如下:

  1. gin_demo
  2. ├── app
  3. ├── blog
  4. ├── handler.go
  5. └── router.go
  6. └── shop
  7. ├── handler.go
  8. └── router.go
  9. ├── go.mod
  10. ├── go.sum
  11. ├── main.go
  12. └── routers
  13. └── routers.go

其中app/blog/router.go用来定义post相关路由信息,具体内容如下:

  1. func Routers(e *gin.Engine) {
  2. e.GET("/post", postHandler)
  3. e.GET("/comment", commentHandler)
  4. }

app/shop/router.go用来定义shop相关路由信息,具体内容如下:

  1. func Routers(e *gin.Engine) {
  2. e.GET("/goods", goodsHandler)
  3. e.GET("/checkout", checkoutHandler)
  4. }

routers/routers.go中根据需要定义Include函数用来注册子app中定义的路由,Init函数用来进行路由的初始化操作:

  1. type Option func(*gin.Engine)
  2. var options = []Option{}
  3. // 注册app的路由配置
  4. func Include(opts ...Option) {
  5. options = append(options, opts...)
  6. }
  7. // 初始化
  8. func Init() *gin.Engine {
  9. r := gin.New()
  10. for _, opt := range options {
  11. opt(r)
  12. }
  13. return r
  14. }

main.go中按如下方式先注册子app中的路由,然后再进行路由的初始化:

  1. func main() {
  2. // 加载多个APP的路由配置
  3. routers.Include(shop.Routers, blog.Routers)
  4. // 初始化路由
  5. r := routers.Init()
  6. if err := r.Run(); err != nil {
  7. fmt.Println("startup service failed, err:%v\n", err)
  8. }
  9. }

转自:https://www.liwenzhou.com/posts/Go/gin_routes_registry/