免费加速器,seo刷关键词排名优化,wordpress 不收录,wordpress多个字体大小文章目录 一、中间件二、Gin路由简介1、普通路由2、路由组 三、路由拆分与注册1、基本的路由注册2、路由拆分成单独文件或包3、路由拆分成多个文件4、路由拆分到不同的APP 一、中间件
在日常工作中#xff0c;经常会有一些计算接口耗时和限流的操作#xff0c;如果每写一个接… 文章目录 一、中间件二、Gin路由简介1、普通路由2、路由组 三、路由拆分与注册1、基本的路由注册2、路由拆分成单独文件或包3、路由拆分成多个文件4、路由拆分到不同的APP 一、中间件
在日常工作中经常会有一些计算接口耗时和限流的操作如果每写一个接口都需要手动的去加上计算耗时和限流的代码显然是很冗余且不好维护的还很容易遗漏。这个时候我们一般会想到使用中间件的方式将这些与业务无关的代码写到中间件去然后安到每个接口中去就行了。
package mainimport (fmtnet/httptimegithub.com/gin-gonic/gin
)func timeMiddleware() gin.HandlerFunc {return func(ctx *gin.Context) {begin : time.Now()defer func() {fmt.Printf(use time %d ms\n, time.Since(begin).Milliseconds())}()ctx.Next()}
}func limitMiddleware() gin.HandlerFunc {// 限流最高并发为10,这里return的func会必闭包使用这个limitChan从而达到限流效果limitChan : make(chan struct{}, 10) return func(ctx *gin.Context) {defer func() {-limitChan}()limitChan - struct{}{}ctx.Next()}
}func bizHandler(ctx *gin.Context) {time.Sleep(100 * time.Millisecond)ctx.String(http.StatusOK, gin 中间件)}func main() {engine : gin.Default()// Use方法就是将中间件放到了链条的首部注意Use接收的是可变参数可接收多个中间件// engine.Use(timeMiddleware(),limitMiddleware())// 如果是分别使用use则要注意一下顺序如这里将timeMiddleware后写//是因为想把timeMiddleware放到链条首部从而将限流中间件的耗时也统计到engine.Use(limitMiddleware())engine.Use(timeMiddleware())engine.GET(/v1, bizHandler)// engine.GET(/v2,timeMiddleware(), bizHandler)engine.Run(127.0.0.1:8080)
}
注意事项
中间件是gin.HandlerFunc类型在使用limitMiddleware和timeMiddleware时我们加了小括号因为它们的返回值才是gin.HandlerFunc类型engine.Get,engine.Post,engine.Use方法接收的都是可变长参数如示例中的v2路径可以直接将中间件对指定的路径使用或者用Use一次全局使用多个中间件使用多个Use时注意使用顺序后使用的Use里面的中间件会放到链表首部如果中间件中没有使用ctx.Next,则是将当前中间件执行完后再去执行链表上的下一个handler如果使用了ctx.Next则表示从此处开始先将链表后面的handler都执行完然后再回溯到这里的ctx.Next位置来继续执行当前中间件函数中的后续代码。
二、Gin路由简介
1、普通路由
r.GET(/index, func(c *gin.Context) {...})
r.GET(/login, func(c *gin.Context) {...})
r.POST(/login, func(c *gin.Context) {...})此外还有一个可以匹配所有请求方法的Any方法如下
r.Any(/test, func(c *gin.Context) {...})为没有配置处理函数的路由添加处理程序默认情况下它返回404代码下面的代码为没有匹配到路由的请求都返回views/404.html页面。
r.NoRoute(func(c *gin.Context) {c.HTML(http.StatusNotFound, views/404.html, nil)})2、路由组
我们可以将拥有共同URL前缀的路由划分为一个路由组。习惯性一对{}包裹同组的路由这只是为了看着清晰用不用{}包裹功能上没什么区别。
func main() {r : gin.Default()userGroup : r.Group(/user){userGroup.GET(/index, func(c *gin.Context) {...})userGroup.GET(/login, func(c *gin.Context) {...})userGroup.POST(/login, func(c *gin.Context) {...})}shopGroup : r.Group(/shop){shopGroup.GET(/index, func(c *gin.Context) {...})shopGroup.GET(/cart, func(c *gin.Context) {...})shopGroup.POST(/checkout, func(c *gin.Context) {...})}r.Run()
}通常我们将路由分组用在划分业务逻辑或划分API版本时。
三、路由拆分与注册
1、基本的路由注册
下面是最基础的gin路由注册方式适用于路由比较少的简单项目或者项目demo。
package mainimport (net/httpgithub.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)}
}2、路由拆分成单独文件或包
当项目的规模增大后就不太适合继续在项目的main.go文件中去实现路由注册相关逻辑了我们会倾向于把路由部分的代码都拆分出来形成一个单独的文件或包
我们在routers.go文件中定义并注册路由信息
package mainimport (net/httpgithub.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\n, err)}
}此时的目录结构
gin_demo
├── go.mod
├── go.sum
├── main.go
└── routers.go一般会把路由部分的代码单独拆分成包的拆分后的目录结构如下
gin_demo
├── go.mod
├── go.sum
├── main.go
└── routers└── routers.gorouters/routers.go
需要注意此时setupRouter需要改成首字母大写因为和main.go已经不在一个包中了要在main.go中调用SetupRouter所以他必须是可导出的
package routersimport (net/httpgithub.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 mainimport (fmtgin_demo/routers
)func main() {r : routers.SetupRouter()if err : r.Run(); err ! nil {fmt.Println(startup service failed, err:%v\n, err)}
}3、路由拆分成多个文件
当我们的业务规模继续膨胀单独的一个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.gorouters/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\n, err)}
}4、路由拆分到不同的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)
}在第三步迭代中3、路由拆分成多个文件我们在main.go中使用了两次routers.LoadXXX(r)事实上他们是同种类型的函数,当这种调用比较多时也是累赘故可以定义option使用函数数选项模式使得代码更优雅。
func main() {r : gin.Default()// 使用了两次routers.LoadXXX(r)routers.LoadBlog(r)routers.LoadShop(r)if err : r.Run(); err ! nil {fmt.Println(startup service failed, err:%v\n, err)}
}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\n, err)}
}