- Created by 郭强, last modified on May 08, 2024
分组路由是业务项目中主要使用的路由注册方式。
分组路由
GoFrame
框架支持分组路由的注册方式,可以给分组路由指定一个prefix
前缀(也可以直接给定/
前缀,表示注册在根路由下),在该分组下的所有路由注册都将注册在该路由前缀下。分组路由注册方式也是推荐的路由注册方式。
接口文档:https://pkg.go.dev/github.com/gogf/gf/v2/net/ghttp#RouterGroup
// 创建分组路由 func (s *Server) Group(prefix string, groups ...func(g *RouterGroup)) *RouterGroup // 注册Method路由 func (g *RouterGroup) ALL(pattern string, object interface{}, params...interface{}) func (g *RouterGroup) GET(pattern string, object interface{}, params...interface{}) func (g *RouterGroup) PUT(pattern string, object interface{}, params...interface{}) func (g *RouterGroup) POST(pattern string, object interface{}, params...interface{}) func (g *RouterGroup) DELETE(pattern string, object interface{}, params...interface{}) ... // 中间件绑定 func (g *RouterGroup) Middleware(handlers ...HandlerFunc) *RouterGroup // 批量注册 func (g *RouterGroup) Map(m map[string]interface{}) func (g *RouterGroup) ALLMap(m map[string]interface{})
简要介绍:
Group
方法用于创建一个分组路由对象,并且支持在指定域名对象上创建。- 以
HTTP Method
命名的方法用于绑定指定的HTTP Method
路由;其中ALL
方法用于注册所有的HTTP Method
到指定的函数/对象/控制器上;REST
方法用于注册RESTful
风格的路由,需给定一个执行对象或者控制器对象。 Middleware
方法用于绑定一个或多个中间件到当前分组的路由上,具体详见中间件章节。
简单示例
package main import ( "github.com/gogf/gf/v2/frame/g" "github.com/gogf/gf/v2/net/ghttp" ) func main() { s := g.Server() s.Group("/api", func(group *ghttp.RouterGroup) { group.ALL("/all", func(r *ghttp.Request) { r.Response.Write("all") }) group.GET("/get", func(r *ghttp.Request) { r.Response.Write("get") }) group.POST("/post", func(r *ghttp.Request) { r.Response.Write("post") }) }) s.SetPort(8199) s.Run() }
执行后,终端打印出路由表如下:
SERVER | DOMAIN | ADDRESS | METHOD | ROUTE | HANDLER | MIDDLEWARE |---------|---------|---------|--------|-----------|-----------------|------------| default | default | :8199 | ALL | /api/all | main.main.func1 | |---------|---------|---------|--------|-----------|-----------------|------------| default | default | :8199 | GET | /api/get | main.main.func2 | |---------|---------|---------|--------|-----------|-----------------|------------| default | default | :8199 | POST | /api/post | main.main.func3 | |---------|---------|---------|--------|-----------|-----------------|------------|
其中,/api/get
仅允许GET
方式访问,/api/post
仅允许POST
方式访问,/api/all
允许所有的方式访问。
我们使用curl
工具来测试一下:
/api/get
$ curl http://127.0.0.1:8199/api/get get $ curl -X POST http://127.0.0.1:8199/api/get Not Found
/api/post
$ curl http://127.0.0.1:8199/api/post Not Found $ curl -X POST http://127.0.0.1:8199/api/post post
/api/all
$ curl http://127.0.0.1:8199/api/all all $ curl -X POST http://127.0.0.1:8199/api/all all $ curl -X DELETE http://127.0.0.1:8199/api/all all $ curl -X OPTIONS http://127.0.0.1:8199/api/all all
层级注册
GoFrame
框架的层级路由注册方式灵感来源于PHP Laravel
框架。
推荐使用路由层级注册方式,注册的路由代码更清晰直观。GoFrame
框架的分组路由注册支持更加直观优雅层级的注册方式,以便于开发者更方便地管理路由列表。路由层级注册方式也是推荐的路由注册方式。我们来看一个比较完整的示例,该示例中注册了使用到了中间件、HOOK
以及不同HTTP Method
绑定的路由注册:
package main import ( "fmt" "net/http" "github.com/gogf/gf/v2/frame/g" "github.com/gogf/gf/v2/net/ghttp" ) func MiddlewareAuth(r *ghttp.Request) { if r.Get("token").String() != "123456" { r.Response.WriteStatus(http.StatusForbidden) return } r.Middleware.Next() } func MiddlewareCORS(r *ghttp.Request) { r.Response.CORSDefault() r.Middleware.Next() } func MiddlewareLog(r *ghttp.Request) { r.Middleware.Next() fmt.Println(r.Response.Status, r.URL.Path) } func main() { s := g.Server() s.Use(MiddlewareLog) s.Group("/api.v2", func(group *ghttp.RouterGroup) { group.Middleware(MiddlewareAuth, MiddlewareCORS) group.GET("/test", func(r *ghttp.Request) { r.Response.Write("test") }) group.Group("/order", func(group *ghttp.RouterGroup) { group.GET("/list", func(r *ghttp.Request) { r.Response.Write("list") }) group.PUT("/update", func(r *ghttp.Request) { r.Response.Write("update") }) }) group.Group("/user", func(group *ghttp.RouterGroup) { group.GET("/info", func(r *ghttp.Request) { r.Response.Write("info") }) group.POST("/edit", func(r *ghttp.Request) { r.Response.Write("edit") }) group.DELETE("/drop", func(r *ghttp.Request) { r.Response.Write("drop") }) }) group.Group("/hook", func(group *ghttp.RouterGroup) { group.Hook("/*", ghttp.HookBeforeServe, func(r *ghttp.Request) { r.Response.Write("hook any") }) group.Hook("/:name", ghttp.HookBeforeServe, func(r *ghttp.Request) { r.Response.Write("hook name") }) }) }) s.SetPort(8199) s.Run() }
执行后,注册的路由列表如下:
SERVER | DOMAIN | ADDRESS | METHOD | ROUTE | HANDLER | MIDDLEWARE |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | ALL | /* | main.MiddlewareLog | GLOBAL MIDDLEWARE |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | ALL | /api.v2/hook/* | main.main.func1.4.1 | HOOK_BEFORE_SERVE |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | ALL | /api.v2/hook/:name | main.main.func1.4.2 | HOOK_BEFORE_SERVE |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | GET | /api.v2/order/list | main.main.func1.2.1 | main.MiddlewareAuth,main.MiddlewareCORS |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | PUT | /api.v2/order/update | main.main.func1.2.2 | main.MiddlewareAuth,main.MiddlewareCORS |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | GET | /api.v2/test | main.main.func1.1 | main.MiddlewareAuth,main.MiddlewareCORS |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | DELETE | /api.v2/user/drop | main.main.func1.3.3 | main.MiddlewareAuth,main.MiddlewareCORS |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | POST | /api.v2/user/edit | main.main.func1.3.2 | main.MiddlewareAuth,main.MiddlewareCORS |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------| default | default | :8199 | GET | /api.v2/user/info | main.main.func1.3.1 | main.MiddlewareAuth,main.MiddlewareCORS |---------|---------|---------|--------|----------------------|---------------------|-----------------------------------------|
批量注册
Map
通过Map
方法可以实现批量的分组路由注册,但是如果是相同的URI
不同的HTTP Method
需要按照路由规范给定HTTP Method
。使用示例:
package main import ( "github.com/gogf/gf/v2/frame/g" "github.com/gogf/gf/v2/net/ghttp" ) func UserGet(r *ghttp.Request) { r.Response.Write("get") } func UserDelete(r *ghttp.Request) { r.Response.Write("delete") } func main() { s := g.Server() s.Group("/api", func(group *ghttp.RouterGroup) { group.Map(g.Map{ "GET: /user": UserGet, "DELETE: /user": UserDelete, }) }) s.SetPort(8199) s.Run() }
AllMap
也可以使用ALLMap
方法实现批量的分组路由注册,通过该方法注册的路由将把路由函数/对象应用到所有的HTTP Method
。使用示例:
s := g.Server() // 前台系统路由注册 s.Group("/", func(group *ghttp.RouterGroup) { group.Middleware(service.Middleware.Ctx) group.ALLMap(g.Map{ "/": api.Index, // 首页 "/login": api.Login, // 登录 "/register": api.Register, // 注册 "/category": api.Category, // 栏目 "/topic": api.Topic, // 主题 "/topic/:id": api.Topic.Detail, // 主题 - 详情 "/ask": api.Ask, // 问答 "/ask/:id": api.Ask.Detail, // 问答 - 详情 "/article": api.Article, // 文章 "/article/:id": api.Article.Detail, // 文章 - 详情 "/reply": api.Reply, // 回复 "/search": api.Search, // 搜索 "/captcha": api.Captcha, // 验证码 "/user/:id": api.User.Index, // 用户 - 主页 }) // 权限控制路由 group.Group("/", func(group *ghttp.RouterGroup) { group.Middleware(service.Middleware.Auth) group.ALLMap(g.Map{ "/user": api.User, // 用户 "/content": api.Content, // 内容 "/interact": api.Interact, // 交互 "/file": api.File, // 文件 }) }) })
Content Menu
- No labels
21 Comments
manong
郭强
不可以
sanrentai
kn1011
可选的注册对象方法路由
phoenix
那个有注解路由吗,怎么用
sanrentai
tigerOoo
同问,分组路由没法设置根路由,访问直接not found
郭强
sanrentai tigerOoo 哥哥些,问题请提
issue
,另外这个问题我用master
分支并没有复现。tigerOoo
测试最新版,无问题,谢了
jack
出现:
./g16.go:12:11: invalid operation: token == "123456" (mismatched types *gvar.Var and untyped string)
./g16.go:26:9: g.Log().Println undefined (type *glog.Logger has no field or method Println)
倪时鸿
第一个报错,修改这个。
小兵张嘎
如何实现 对象的注册 下面类似的 Init 功能呢?
JCJC错别字检测
本页面,层级注册的例子中,有一处代码:
token == "123456"
编辑器提示:Invalid operation: token == "123456" (mismatched types *gvar.Var and string)
本页面,还有一处代码:
koreanafter
2024年了,这个错还在报
小陈
虽然功能好多好多. 不过真的牛掰. 代码质量比gin好. 也易懂
ceroot
对路由分组批量注册ALLMap在v2版本出现一个问题,有个疑问,如下:
输出版本:
gf -v
GoFrame CLI Tool v2.5.6, https://goframe.org
GoFrame Version: v2.5.6 in current go.mod
批量注册代码:
skyonsky
Before80
$ curl -X POST http://127.0.0.1:8199/api/post post
Before80
在层级注册中有一句:“我们来看一个比较完整的示例,该示例中注册了使用到了中间件、
HOOK
以及不同HTTP Method
绑定的路由注册:”, 语句不是很通顺,觉得可以修改成“我们来看一个比较完整的示例,该示例是一个使用到了中间件、HOOK
以及不同HTTP Method
绑定的路由注册:”, 即将“中注册了”修改成“是一个”。叮当猫
group.REST如何使用?按照绑定对象的方式写了代码,但是没任何效果。
廖国勇
刚开始接触golang,还不太熟悉,准备对*ghttp.Request,扩展两个方法 Success和Fail,用于API里边返 回数据给客户端使用,请问该如何操作。