• gin框架路由拆分与注册


    转自

    gin框架路由拆分与注册

    基本的路由注册

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

    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
    ", err)
    	}
    }
    

      

    路由拆分成单独文件或包

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

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

    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函数:

    func main() {
    	r := setupRouter()
    	if err := r.Run(); err != nil {
    		fmt.Println("startup service failed, err:%v
    ", 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需要改成首字母大写:

    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文件内容如下:

    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
    ", err)
    	}
    }
    

      

    路由拆分成多个文件

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

    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相关的路由注册到指定的路由器:

    func LoadShop(e *gin.Engine)  {
    	e.GET("/hello", helloHandler)
      e.GET("/goods", goodsHandler)
      e.GET("/checkout", checkoutHandler)
      ...
    }
    

      

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

    func LoadBlog(e *gin.Engine) {
    	e.GET("/post", postHandler)
      e.GET("/comment", commentHandler)
      ...
    }
    

      

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

    func main() {
    	r := gin.Default()
    	routers.LoadBlog(r)
    	routers.LoadShop(r)
    	if err := r.Run(); err != nil {
    		fmt.Println("startup service failed, err:%v
    ", 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相关路由信息,具体内容如下:

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

      

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

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

      

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

    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中的路由,然后再进行路由的初始化:

    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
    ", err)
    	}
    }
    

      

    总结

    gin框架是一个非常容易扩展的web框架,本文是我在日常编码中总结的一点点经验,因为世界上不可能有完全相同的项目,每个人也都有自己的编程习惯,关于gin框架路由注册的方式我就在此抛砖引玉了。

     
    Songzhibin
  • 相关阅读:
    使用require.context引入模块
    npm link的使用
    mysql 链接远程数据库
    微信错误:errcode=45015, errmsg=response out of time limit or subscription is canceled
    微信公众号发送模板消息
    VS CODE 开发php实现断点调试
    uni 蓝牙 安卓 监听不到返回数据 不可写入
    vue3.0 兄弟组件传值
    二叉查找树的实现和删除
    模块二:ES新特性与TypeScript、JS性能优化
  • 原文地址:https://www.cnblogs.com/binHome/p/13411822.html
Copyright © 2020-2023  润新知