gin02:gin路径中的参数

本文来探讨gin是如何获取路径中的参数

gin获取路径中参数的示例

go 复制代码
func main() {
    router := gin.Default()

    router.GET("/user/:name", func(c *gin.Context) {
       name := c.Param("name")
       c.String(http.StatusOK, "hello %s", name)
    })
    router.GET("/user/:name/*action", func(context *gin.Context) {
       name := context.Param("name")
       action := context.Param("action")
       meesage := name + " is " + action
       context.String(http.StatusOK, meesage)
    })
    router.POST("/user/:name/*action", func(context *gin.Context) {
       b := context.FullPath() == "/user/:name/*action"
       context.String(http.StatusOK, "%t", b)
    })

    router.GET("/user/groups", func(context *gin.Context) {
       context.String(http.StatusOK, "可用的用户组是[...]")
    })
    router.Run("localhost:9090")
}

gin能够通过context.Param()就非常简单的获取了路径中的参数,接下来探讨它是怎么做到的。

Params

gin.Context的结构体中有一个Params字段,其实际是[] Param类型

bash 复制代码
type Params []Param

Param的结构体十分简单

go 复制代码
type Param struct {
    Key   string
    Value string
}

这就是底层参数的存储的数据结构,在调用context.Param()获取参数的时

go 复制代码
func (c *Context) Param(key string) string {
    return c.Params.ByName(key)
}

实际是调用了c.Params.ByName()

go 复制代码
func (ps Params) ByName(name string) (va string) {
    va, _ = ps.Get(name)
    return
}

而其底层是通过Get获取value

go 复制代码
func (ps Params) Get(name string) (string, bool) {
    for _, entry := range ps {
       if entry.Key == name {
          return entry.Value, true
       }
    }
    return "", false
}

这里可以看出,实际获取参数的操作,就是循环遍历了一遍底层的Param切片。默认返回空字符串""

Params的生命周期

在探讨了底层是如何获得参数的问题之后,现在有一个新的问题,Params的生命周期历程。 1.Params的重置

  • EngineServeHTTP方法中,从Context池中创建或者拿出一个Context
  • 调用Context.reset()针对它进行初始化
ini 复制代码
func (c *Context) reset() {
    c.Writer = &c.writermem
    c.Params = c.Params[:0]
    c.handlers = nil
    c.index = -1

    c.fullPath = ""
    c.Keys = nil
    c.Errors = c.Errors[:0]
    c.Accepted = nil
    c.queryCache = nil
    c.formCache = nil
    c.sameSite = 0
    *c.params = (*c.params)[:0]
    *c.skippedNodes = (*c.skippedNodes)[:0]
}
  1. Params的设置
    • 在根据请求方法获取到对应的路由树之后
    • 调用root.getValue(),传入请求的路径,和c.params
    • 获得一个nodeValue对象,其中包括逻辑处理链参数注册的路由
    • nodeValueparams赋给Context.Params
csharp 复制代码
root := t[i].root
// Find route in tree
value := root.getValue(rPath, c.params, c.skippedNodes, unescape)
if value.params != nil {
    c.Params = *value.params
}
  1. Context被回收
相关推荐
beata2 小时前
Java基础-16:Java内置锁的四种状态及其转换机制详解-从无锁到重量级锁的进化与优化指南
java·后端
Mintopia2 小时前
软件系统中的订单-审核业务架构分析与实践
后端·架构
极客老墨2 小时前
Go创造者Rob Pike带你深入理解 Go 字符串:字节、符文与 Unicode 编码
go
茶杯梦轩2 小时前
从零起步学习RabbitMQ || 第二章:RabbitMQ 深入理解概念 Producer、Consumer、Exchange、Queue 与企业实战案例
服务器·后端·消息队列
随逸1772 小时前
《 吃透RAG:从原理到LangChain实战,彻底解决大模型幻觉问题》
后端
SimonKing2 小时前
觅得又一款轻量级数据库管理工具:GoNavi
java·后端·程序员
小码哥_常2 小时前
面试必知!Java线程池深度剖析
后端
Moment2 小时前
Cursor 的 5 种指令方法比较,你最喜欢哪一种?
前端·后端·github
IT_陈寒2 小时前
Vite快得离谱?揭秘它比Webpack快10倍的5个核心原理
前端·人工智能·后端