html获取url后面的参数_Golang Gin 实战(四)| URL查询参数的获取和原理分析

在 上一篇 Golang Gin 实战(三)| 路由参数 文章中,主要介绍了路由通配符、路由参数,让我们有了一种可以从URL路径中获取参数的方式,同时又不是重复的注册相似的路由。

这一篇,主要介绍查询参数,以及获取查询参数的示例以及底层原理。

查询参数(query parames)

Querystring parameters ,翻译成中文我只能叫查询参数了,不过觉得挺别捏的。其实这种参数我们并不陌生,比如:

https://www.flysnow.org/search?q=golang&sitesearch=https%3A%2F%2Fwww.flysnow.org

URL查询参数,或者也可以简称为URL参数,是存在于我们请求的URL中,以?为起点,后面的k=v&k1=v1&k2=v2这样的字符串就是查询参数,比如我上面示例中的:

?q=golang&sitesearch=https%3A%2F%2Fwww.flysnow.org

这个示例中有两个查询参数键值对:

q=golang
sitesearch=https%3A%2F%2Fwww.flysnow.org

第一个key是q,对应的值是golang。第二个key是sitesearch,对应的值是https%3A%2F%2Fwww.flysnow.org,它们通过&相连。在URL中,多个查询参数键值对通过&相连。

Gin获取查询参数

Gin中,为我们提供了简便的方法来获取查询参数的值,我们只需要知道查询参数的key(参数名)就可以了。

func main() {
    r := gin.Default()

    r.GET("/", func(c *gin.Context) {
        c.String(200, c.Query("wechat"))
    })
    r.Run(":8080")
}

我们运行这段代码,打开浏览器访问http://localhost:8080/?wechat=flysnow_org,就可以看到flysnow_org文字。这表示我们通过c.Query("wechat")获取到了查询参数wechat的值是flysnow_org

Query方法为我们提供了获取对应key的值的能力,如果该key不存在,则返回""字符串。如果对于一些数字参数,比如id如果返回为空的话,我们进行字符串转数字的时候会报错,这时候,我们就可以通过DefaultQuery方法指定一个默认值:

c.DefaultQuery("wechat", "flysnow_org")

c.DefaultQuery("id", "0")

比如这样,尤其是第二个例子,默认为0,让我们字符串转数字很方便。

func (c *Context) Query(key string) string {
    value, _ := c.GetQuery(key)
    return value
}


func (c *Context) DefaultQuery(key, defaultValue string) string {
    if value, ok := c.GetQuery(key); ok {
        return value
    }
    return defaultValue
}

看下这两个函数的源代码实现,它们都是调用的GetQuery方法获取对应的值,唯一不同的是DefaultQuery会判断对应的key是否存在,如果不存在的话,则返回默认defaultValue值。

原理解析

从以上两个获取查询参数值的方法可以看到,他们调用的都是GetQuery,这也是gin.Context的一个方法,它和Query唯一不同的是,它返回两个值,可以告诉我们要获取的key是否存在。

value, ok := c.GetQuery("wechat")

如果我们自己的业务中,需要这类功能,可以用GetQuery来代替Query方法。

GetQuery方法的底层实现其实是c.Request.URL.Query().Get(key),通过url.URL.Query()来获取所有的参数键值对。

本质上是调用的GetQueryArray,取的数组中第一个值
func (c *Context) GetQuery(key string) (string, bool) {
    if values, ok := c.GetQueryArray(key); ok {
        return values[0], ok
    }
    return "", false
}


func (c *Context) GetQueryArray(key string) ([]string, bool) {
    c.getQueryCache() //缓存所有的键值对
    if values, ok := c.queryCache[key]; ok && len(values) > 0 {
        return values, true
    }
    return []string{}, false
}

func (c *Context) getQueryCache() {
    if c.queryCache == nil {
        c.queryCache = c.Request.URL.Query()
    }
}

从以上的实现代码中,可以看到最终的实现都在GetQueryArray方法中,找到对应的key就返回对应的[]string,返回就返回空数组。

这里Gin进行了优化,通过缓存所有的键值对,提升代码的查询效率。这里缓存的queryCache本质上是url.Values,也是一个map[string][]string

type Values map[string][]string

其中c.Request.URL.Query()这个方法就是把?k=v&k1=v1&k2=v2这类查询键值对转换为map[string][]string,所以还是很耗性能的,这里Gin采用了缓存的做法提高了性能挺好,这也是Gin成为性能最快的Golang Web 框架的原因之一吧。

小结

在原理分析这一小结,相信大家也看到了GetQueryArray函数,可以让我们获取key对应的数组,比如多选这类功能,本来打算这一篇文章就介绍下呢,发现这篇内容已经不少了,太多了,大家不容易看完,所以就放到下一节吧。

可能大家没有留意,我的文章,我都会稍微加入一些原理、源代码的分析,这个主要就是为了让大家更了解其中的原理,可以更好的使用它们,更为了大家能够通过这个分析,提升自己的技术能力,比如这里用到的缓存机制。

精彩文章推荐

Golang Gin 实战(三)| 路由参数

Golang Gin 实战(二)| 简便的Restful API 实现

Golang Gin 实战(一)| 快速安装入门

我有几个的Go语言交流微信群,可以扫码关注公众号flysnow_org或者网站 https://www.flysnow.org/,加我好友,我拉你进来。

e9a1b0bfea5d6ba4f06c6d8e30ca0373.png
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值