Gin.RouterGroup:分组、中间件与路径组合

在 Gin 框架中,真正支撑路由能力的不是 gin.Engine,而是 gin.RouterGroup

gin.Engine 本质上是嵌入了 RouterGroup 的增强结构(type Engine struct { *RouterGroup ... }),因此你调用 r.GET()r.Group() 时,实际上是在操作一个 RouterGroup 实例。

RouterGroup 是 Gin 的核心抽象,它提供了:

  • 路径前缀管理(自动拼接)
  • 中间件作用域隔离
  • 嵌套分组能力
  • 统一的方法注册接口(GET/POST/Use/Static 等)

理解 RouterGroup 的工作方式,是掌握 Gin 架构设计的关键

HTTP 方法路由

go 复制代码
// 所有 HTTP 方法

GET(path string, handlers ...HandlerFunc) IRoutes

POST(path string, handlers ...HandlerFunc) IRoutes

PUT(path string, handlers ...HandlerFunc) IRoutes

DELETE(path string, handlers ...HandlerFunc) IRoutes

PATCH(path string, handlers ...HandlerFunc) IRoutes

HEAD(path string, handlers ...HandlerFunc) IRoutes

OPTIONS(path string, handlers ...HandlerFunc) IRoutes

ANY(path string, handlers ...HandlerFunc) IRoutes

  

// 自定义方法

Handle(httpMethod, relativePath string, handlers ...HandlerFunc) IRoutes

中间件管理

go 复制代码
// 添加中间件
Use(middleware ...HandlerFunc) IRoutes

分组创建

go 复制代码
// 创建子分组
Group(component string, handlers ...HandlerFunc) *RouterGroup

静态文件服务

go 复制代码
// 静态文件
StaticFile(relativePath, filepath string) IRoutes
Static(relativePath, root string) IRoutes
StaticFS(relativePath string, fs http.FileSystem) IRoutes

路径组合规则

go 复制代码
// 基础路径组合规则
r := gin.Default()
api := r.Group("/api")           // basePath: "/api"
v1 := api.Group("/v1")           // basePath: "/api/v1"  
users := v1.Group("/users")      // basePath: "/api/v1/users"

// 最终路由:
// users.GET("/profile") → 实际路径: /api/v1/users/profile
// users.POST("/create") → 实际路径: /api/v1/users/create

基础用法

go 复制代码
func main() {
    r := gin.Default()
    
    // Engine 本身就是一个 RouterGroup
    r.GET("/", handler)  // r 是 *RouterGroup
    
    // 创建分组
    api := r.Group("/api")  // 返回 *RouterGroup
    api.GET("/users", handler)
    api.POST("/users", handler)
}

分组嵌套

go 复制代码
func main() {
    r := gin.Default()
    
    // 一级分组
    api := r.Group("/api")
    {
        api.GET("/users", handler1)
        api.POST("/users", handler2)
        
        // 二级分组
        v1 := api.Group("/v1")
        {
            v1.GET("/users", handler3)
            v1.POST("/users", handler4)
        }
        
        v2 := api.Group("/v2")
        {
            v2.GET("/users", handler5)
        }
    }
}

中间件分组

go 复制代码
func main() {
    r := gin.Default()
    
    // 公共路由组(无中间件)
    public := r.Group("/public")
    {
        public.GET("/info", handler1)
    }
    
    // 受保护路由组(带认证中间件)
    protected := r.Group("/api")
    protected.Use(AuthMiddleware())  // 该组所有路由都使用认证中间件
    {
        protected.GET("/profile", handler2)
        protected.POST("/posts", handler3)
        
        // 在分组内再创建子分组
        admin := protected.Group("/admin")
        admin.Use(AdminMiddleware())  // 管理员中间件
        {
            admin.GET("/dashboard", handler4)
            admin.DELETE("/users/:id", handler5)
        }
    }
}

完整示例

go 复制代码
package main

import (
    "github.com/gin-gonic/gin"
    "net/http"
)

func AuthMiddleware() gin.HandlerFunc {
    return func(c *gin.Context) {
        // 简单认证逻辑
        token := c.GetHeader("Authorization")
        if token == "" {
            c.JSON(401, gin.H{"error": "Unauthorized"})
            c.Abort()
            return
        }
        c.Next()
    }
}

func AdminMiddleware() gin.HandlerFunc {
    return func(c *gin.Context) {
        // 简单管理员验证
        role := c.GetHeader("Role")
        if role != "admin" {
            c.JSON(403, gin.H{"error": "Forbidden"})
            c.Abort()
            return
        }
        c.Next()
    }
}

func main() {
    r := gin.Default()
    
    // 根分组(Engine 本身)
    r.GET("/", func(c *gin.Context) {
        c.JSON(200, gin.H{"message": "Root path"})
    })
    
    // API 分组
    api := r.Group("/api")
    {
        api.GET("/health", func(c *gin.Context) {
            c.JSON(200, gin.H{"status": "ok"})
        })
        
        // 用户相关接口
        users := api.Group("/users")
        users.Use(AuthMiddleware())  // 用户接口需要认证
        {
            users.GET("/", func(c *gin.Context) {
                c.JSON(200, gin.H{"users": []string{"user1", "user2"}})
            })
            users.GET("/:id", func(c *gin.Context) {
                id := c.Param("id")
                c.JSON(200, gin.H{"id": id, "name": "John"})
            })
            users.POST("/", func(c *gin.Context) {
                c.JSON(201, gin.H{"message": "User created"})
            })
        }
        
        // 管理员接口
        admin := api.Group("/admin")
        admin.Use(AuthMiddleware(), AdminMiddleware())  // 需要认证+管理员权限
        {
            admin.GET("/dashboard", func(c *gin.Context) {
                c.JSON(200, gin.H{"message": "Admin dashboard"})
            })
            admin.GET("/users", func(c *gin.Context) {
                c.JSON(200, gin.H{"message": "Admin users list"})
            })
        }
    }
    
    // 静态文件
    static := r.Group("/static")
    static.Static("/files", "./files")  // 静态文件服务
    
    r.Run()
}
相关推荐
我是李龙9 小时前
第二十六章 全链路监控体系: 服务器资源、中间件状态与业务指标的可视化大屏开发
中间件
别抢我的锅包肉1 天前
FastAPI + Vue3 + Vite 跨域报错全解:从 `Access-Control-Allow-Origin missing` 到彻底修复
中间件·状态模式·fastapi
爱吃烤鸡翅的酸菜鱼2 天前
Java 事件发布-订阅机制全解析:从原生实现到主流中间件
java·中间件·wpf·事件·发布订阅
my_styles2 天前
linux系统下安装 tengine / 宝兰德等国产信创中间件和闭坑
linux·运维·服务器·spring boot·nginx·中间件
ZHENGZJM2 天前
Gin 鉴权中间件设计与实现
中间件·gin
ZHENGZJM2 天前
认证增强:图形验证码、邮箱验证与账户安全
安全·react.js·go·gin
开心码农1号2 天前
mq是什么,常用mq的使用场景有哪些?
中间件·rabbitmq
斌味代码2 天前
Next.js 14 App Router 完全指南:服务端组件、流式渲染与中间件实战
开发语言·javascript·中间件
女王大人万岁3 天前
Golang实战gin-swagger:自动生成API文档
服务器·开发语言·后端·golang·gin
so2F32hj23 天前
一款Go语言Gin框架DDD脚手架,适合快速搭建项目
开发语言·golang·gin