Featured image of post Gin框架从入门到实战:核心用法与最佳实践

Gin框架从入门到实战:核心用法与最佳实践

Gin 凭借其高性能和简洁设计,成为 Go 语言 Web 开发的首选框架。本文从基础用法到高级特性,覆盖了路由、中间件、参数绑定等核心内容。掌握这些技能后,可尝试结合 gRPC、Swagger 等工具构建企业级应用。

为什么选择Gin框架?

Gin 是一个基于 Go 语言的高性能 Web 框架,具备以下优势:

  • 轻量高效:底层依赖 net/http,性能接近原生。
  • 简洁优雅:API 设计友好,支持路由分组、中间件链、参数绑定等特性。
  • 生态丰富:内置 JSON 解析、日志记录、错误恢复等实用功能,社区插件生态完善。
  • 无论是构建 RESTful API 还是全栈应用,Gin 都能显著提升开发效率。

安装

要安装Gin软件包,您需要安装Go并首先设置Go工作区。

  • 首先需要安装Go(需要1.10+版本),然后可以使用下面的Go命令安装Gin。
1
go get -u github.com/gin-gonic/gin
  • 将其导入您的代码中:
1
import "github.com/gin-gonic/gin"

基础示例

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
package main

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

func main() {
    // 1.创建 (实例化gin.Engine结构体对象)
    r := gin.Default()
    // 2.绑定路由规则,执行的函数
    // gin.Context,封装了request和response
    r.GET("/", func(c *gin.Context) {
        c.String(http.StatusOK, "Hello, Gin!")
    })
    // 3.监听端口,默认在8080
    //  Run("里面不指定端口号默认为8080")
    r.Run(":8080")
}

运行后访问 http://localhost:8080,即可看到返回的字符串。

Gin工工作作流流程

核心概念

  • Engine 容器对象,整个框架的基础
  • Engine.tree 负责存储路由和handle方法的映射,采用类似于字典树的结构
  • Engine.RouterGroup 其中handlers存储着所有中间件
  • Context 上下文对象,负责处理 请求回应 ,其中handles的存储处理请求时中间件和处理方法的

在这里插入图片描述

请求处理 流程

在这里插入图片描述

GIN启动流程

1
Gin初始化----> Use 中间件  ---->  注册Routers路由 ----> RUN()启动

Gin原原理解析

参考资料:http://v5blog.cn/pages/dd7d5a/

gin.Default()

Default()跟New()几乎一模一样, 就是调用了gin内置的Logger(), Recovery()中间件

1
2
3
4
5
6
7
8
// Default返回一个已经附加了Logger和Recovery中间件的Engine实例
func Default() *Engine {
	debugPrintWARNINGDefault()
	engine := New() "# 默认实例
	// 注册中间建,中间件的是一个函数,最终只要返回一个 type HandlerFunc func(*Context) 就可以
	engine.Use(Logger(), Recovery()) // 默认注册的两个中间件
	return engine
}

engine := New() 初初始化始化

通过调用 gin.New() 方法来实例化 Engine容器

engine.Use() 注注册册中间件 中间件

在这里插入图片描述

路由与参数处理

无参路由

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
func HelloWorldhandler(ctx *gin.Context){
    ctx.JSON(200, gin.H{
        "message": "Hello World",
    })
}

func main() {
    // 创建一个默认的路由器
    router := gin.Default()
    // gin.Context是一个结构体,包含了请求和响应的细节, 封装request和response
    router.GET("/",HelloWorldhandler)
    // 路由重定向
    router.GET("/re", func(c *gin.Context) {
        c.Redirect(http.StatusMovedPermanently, "https://www.baidu.com")
    })
    // 启动服务器
    router.Run(":8081")
}

动态路由参数

通过 :param 捕获 URL 中的变量: (可以通过Context的Param方法来获取API参数)

1
2
3
4
r.GET("/book/:id", func(c *gin.Context) {
    bookID := c.Param("id")
    c.String(http.StatusOK, "书籍ID: %s", bookID)
})

查询参数

使用 Query 或 DefaultQuery 获取 URL 参数: http://127.0.0.1:8000/user?name=zhangsan

1
2
3
4
5
r.GET("/user", func(c *gin.Context) {
    name := c.Query("name")
    role := c.DefaultQuery("role", "guest")
    c.JSON(200, gin.H{"name": name, "role": role})
})

ShouldBind参数绑定

通过 ShouldBind 自动解析请求体(支持 JSON、Form 等):

我们可以基于请求的 Content-Type 识别请求数据类型并利用反射机制 自动提取请求中 QueryString 、 form表单 、 JSON 、 XML 等参数到结构体中

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
type LoginForm struct {
    Username string `form:"username" binding:"required"`
    Password string `form:"password" binding:"required"`
}

r.POST("/login", func(c *gin.Context) {
    var form LoginForm
    if err := c.ShouldBind(&form); err != nil {
        c.JSON(400, gin.H{"error": err.Error()})
        return
    }
    c.String(200, "登录成功: %s", form.Username)
})

完整代码案例

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
package main

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

// HelloWorldhandler 无参数路由处理函数
func HelloWorldhandler(ctx *gin.Context){
    ctx.JSON(200, gin.H{
        "message": "Hello World",
    })
}

func GetBookDetailHandler(ctx *gin.Context )  {
    bookId := ctx.Param("id")

    ctx.String(http.StatusOK, fmt.Sprintf("成功获取书籍详情:%s", bookId))

}

func GetUserDetailHandlers(ctx *gin.Context)   {
    username := ctx.Query("username")
    ctx.String(http.StatusOK, fmt.Sprintf("成功获取用户详情:%s", username))
}

type Login struct {
    Username string `form:"username" json:"username" binding:"required"`
    Password string `form:"password" json:"password" binding:"required"`
}

func ResponseJsonHandler(c *gin.Context) {
    type Data struct {
        Msg string `json:"msg:"`
        Code int    `json:"code"`
    }
    d := Data{
        Msg: "success",
        Code: 200,
    }
    c.JSON(http.StatusOK, d)
}
func Loginhandler(c *gin.Context)  {
    var login Login
    if err := c.ShouldBind(&login); err == nil {
        c.JSON(http.StatusOK, gin.H{
            "username": login.Username,
            "password": login.Password,
        })
    } else {
        c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
    }
}

func ResponseStringHandle(c *gin.Context)  {
    c.String(http.StatusOK, "Hello World")
}

func main() {
    // 创建一个默认的路由器
    router := gin.Default()
    // gin.Context是一个结构体,包含了请求和响应的细节, 封装request和response
    // 无参数路由
    router.GET("/",HelloWorldhandler)
    // 返回字符串
    router.GET("/string", ResponseStringHandle)
    // 返回json
    router.GET("/json", ResponseJsonHandler)
    // 动态路由参数
    router.GET("/book/:id", GetBookDetailHandler)
    // 查询参数 Query 或 DefaultQuery
    router.GET("/user/", GetUserDetailHandlers)
    // 参数绑定
    router.POST("/login", Loginhandler)
    // 路由重定向
    router.GET("/re", func(c *gin.Context) {
        c.Redirect(http.StatusMovedPermanently, "https://www.baidu.com")
    })
    // 启动服务器
    router.Run(":8081")
}

路由分发

为什么需要路由分发?

  • 我们一个项目有非常多的模块,如果全部写在一块导致代码结构混乱,不利于后续的扩展
  • 按照大的模块,每个模块有自己独立的路由,主路由可以再main.go中进行注册

项目结构

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

main.go

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15

import (
"days/routers"
"fmt"
"github.com/gin-gonic/gin"
)
func main() {
    router := gin.Default()
    // 全局中间件
    router.Use(MiddleWare())
    // 加载路由
    routers.LoadUsers(router)
    routers.LoadBooks(router)
    router.Run(":8081")
}

routers/users.go

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
package routers
import (
    "fmt"
    "github.com/gin-gonic/gin"
    "net/http"
    "time"
)

func LoadUsers(e *gin.Engine)  {
    e.GET("/user",MiddleWareOne(),UserHandler)
}

func UserHandler(c *gin.Context) {

    fmt.Println("我是用户路由")
    time.Sleep(time.Second * 5)
    c.JSON(http.StatusOK,gin.H{
        "message" : "weclome user",
    })
}

routers/books.go

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
package routers
import (
"net/http"
"github.com/gin-gonic/gin"
)
func LoadBooks(e *gin.Engine) {
    e.GET("/book", GetBookHandler)
}
func GetBookHandler(c *gin.Context) {
    c.JSON(http.StatusOK, gin.H{
        "message": "Book Router",
    })
}

在这里插入图片描述

中间件

在这里插入图片描述

  • Gin框架允许开发者在处理请求的过程中,加入用户自己的钩子(Hook)函数。
  • 这个钩子函数就叫中间件,中间件适合处理一些公共的业务逻辑
  • 比如登录认证、权限校验、数据分页、记录日志、耗时统计等

全局中间件

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
func Logger() gin.HandlerFunc {
    return func(c *gin.Context) {
        start := time.Now()
        c.Next()
        latency := time.Since(start)
        fmt.Printf("请求耗时: %v\n", latency)
    }
}

func main() {
    r := gin.Default()
    r.Use(Logger()) // 全局生效
    r.GET("/", func(c *gin.Context) { /* ... */ })
}

局部中间件

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
func AuthMiddleware() gin.HandlerFunc {
    return func(c *gin.Context) {
        token := c.GetHeader("token")
        if token != "SECRET_KEY" {
            c.AbortWithStatusJSON(401, gin.H{"error": "身份验证失败"})
        }
        c.Next()
    }
}

r.GET("/profile", AuthMiddleware(), func(c *gin.Context) {
    c.JSON(200, gin.H{"data": "用户信息"})
})

next()方法

  • 在中间件中调用next()方法,会从next()方法调用的地方跳转到Handler函数
  • Handler函数执行完成,若中间件还有部分代码未执行(中间件中next()之后的代码),则执行该代码 在这里插入图片描述
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
package main

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

func main() {
    r := gin.Default()
    r.Use(Log(), RequestID())
    r.GET("/", func(c *gin.Context) {
        fmt.Println("app running  1")
        time.Sleep(time.Second * 5)
        c.String(http.StatusOK, "hello World!")
    })
    r.Run()
}
func Log() gin.HandlerFunc {
    return func(c *gin.Context) {
        fmt.Println("log start")
        c.Next()
        fmt.Println("log end")
    }
}
func RequestID() gin.HandlerFunc {
    return func(c *gin.Context) {
        fmt.Println("requestid start")
        c.Next()
        fmt.Println("requestid end")
    }
}

实现token认证

  • http://127.0.0.1:8080/index index首页无需token直接访问
  • http://127.0.0.1:8080/home home家目录需要对token进行验证,验证通过才可访问
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
package main

import (
    "fmt"
    "github.com/gin-gonic/gin"
)

func AuthMiddleWare() func(c *gin.Context)  {

    return func(c *gin.Context){
        // 客户端携带token 有三种方式 1.放在请求头 2.放在请求体 3.放在url
        // token 验证成功,返回 c.next()才会继续,否则 c.Abort()
        token := c.Request.Header.Get("token")
        fmt.Println("获取token:",token)
        if token == "" {
            c.JSON(200, gin.H{
                "code": 401,
                "msg": "身份验证不通过",
            })
            c.Abort()
        }
        if token != "123456" {
            c.JSON(200, gin.H{
                "code": 401,
                "msg": "token错误",
            })
            c.Abort()
        }

    }
}





func main()  {
    router := gin.Default()
    // 首页无需验证
    router.GET("/", func(c *gin.Context) {
        c.JSON(200, gin.H{
            "message": "首页",
        })
    })
    // Home页面需要验证
    router.GET("/home", AuthMiddleWare(), func(c *gin.Context) {
        c.JSON(200, gin.H{
            "message": "Home页面",
        })
    })
    router.Run(":8081")
}

在这里插入图片描述

未来的你,会感谢今天仍在努力奋斗的你