首页 > 其他分享 >gin.context学习

gin.context学习

时间:2024-11-26 10:01:10浏览次数:5  
标签:map return string Bind 学习 参数 context gin 函数

gin.context是一个结构体类型,其定义如下:
type Context struct {

// 定义了一些私有成员变量,用于存储请求和响应等信息
writermem responseWriter

Request   *http.Request  // 保存request请求

Writer    ResponseWriter // 回写response 

Params   Params

handlers  HandlersChain  // 该次请求所有的中间件函数和处理函数

index     int8           // HandlersChain的下标,用来调用某个具体的HandlerFunc

fullPath  string         // 请求的url路径

engine    *Engine        // 对server Engine的引用

Keys      map[string]any // 用于上下游之间传递参数

params       *Params

skippedNodes *[]skippedNode

mu sync.RWMutex

// Keys is a key/value pair exclusively for the context of each request.
Keys map[string]any

//Errors is a list of errors attached to all the handlers/middlewares who used this context.
Errors errorMsgs

// Accepted defines a list of manually accepted formats for content negotiation.
Accepted []string

// queryCache caches the query result from c.Request.URL.Query().
queryCache url.Values

// formCache caches c.Request.PostForm, which contains the parsed form data from POST, PATCH,
// or PUT body parameters.
formCache url.Values

// SameSite allows a server to define a cookie attribute making it impossible for
// the browser to send this cookie along with cross-site requests.
sameSite http.SameSite

}

中间件:
Context 中封装了原生的 Go HTTP 请求和响应对象,同时还提供了一些处理请求和生成响应的各种方法,用于获取请求和响应的信息、设置响应头、设置响应状态码等操作.在Gin中,Context 是通过中间件来传递的,在处理 HTTP 请求时,Gin 会依次执行注册的中间件,每个中间件可以对 Context 进行一些操作,然后将 Context 传递给下一个中间件。
例如,下面是一个简单的中间件,用于在请求头中设置一个自定义的 X-Request-ID:

func RequestIDMiddleware() gin.HandlerFunc {
return func(c *gin.Context) {
requestID := generateRequestID()
c.Request.Header.Set("X-Request-ID", requestID)
c.Next() // 将 Context 传递给下一个中间件
}
}
在上面的中间件中,生成了一个请求 ID,然后将其设置到请求头中,接着,调用 c.Next() 方法将Context 传递给下一个中间件,这样,下一个中间件就可以通过 c.Request.Header.Get("X-Request-ID") 获取到这个请求 ID

从上面可以知道:一次请求的所有中间件函数和请求处理函数都在Context.handlers中,因此,当请求到来时,只需要依次调用Context.handlers中的所有HandlerFunc即可,这就是调用中间件中的一个最重要的函数Next(),定义如下:

func (c *Context) Next() {
c.index++  // 依次遍历所有的中间件函数,并调用他们
for c.index < int8(len(c.handlers)) {
c.handlersc.index
c.index++
}
}
除了顺序执行所有中间件,还要有在某个中间件函数中终止处理的能力,比如某个中间件负责权限校验,如果用户的校验没通过,直接返回Not Authorized,跳过后续的处理,这个是通过Abort函数实现的:

func (c *Context) Abort() {
c.index = abortIndex // abortIndex是个常量=63
}
abort()的原理非常简单:直接让c.Index等于最大值,这样剩余的中间件函数都没机会执行,从这个函数中可以看出,中间件的数量是有上限的,上限就是63个

参数传递:
Context中有成员Keys,上游需要传递的变量可以放在里面,下游处理时再从里面取出来,实现上下游参数传递,对应Set()和Get()方法

func (c *Context) Set(key string, value any) {
c.mu.Lock() // 用来保护c.Keys并发安全
if c.Keys == nil {
c.Keys = make(map[string]any)
}

c.Keys[key] = value
c.mu.Unlock()

}

func (c *Context) Get(key string) (value any, exists bool) {
c.mu.RLock()
value, exists = c.Keys[key]
c.mu.RUnlock()
return
}

Get()函数还有很多衍生函数,比如GetString,GetStringMapString,GetStringMapStringSlice等,都是在Get的基础上,将数据转换成我们需要的类型再返回

参数绑定:
请求放到Context.Request里,需要解析成结构体或map,才好在业务代码里使用,这一部分是通过Bind()系列函数实现的,Bind()系列函数的作用就是根据request的数据类型,将其解析到结构体或map里,简单的看一个参数绑定的实现:

func (c *Context) ShouldBindWith(obj any, b binding.Binding) error {
return b.Bind(c.Request, obj) // 底层调用的是binding相关的函数
}
针对不同的请求,gin提供了很多函数,用于解析对应的参数,常用的函数如下:

Param(key string) string // 用于获取url参数,比如/welcome/:user_id中的user_id

// 获取GET请求中携带的参数
GetQueryArray(key string) ([]string, bool)
GetQuery(key string)(string, bool)
Query(key string) string
DefaultQuery(key, defaultValue string) string

// 获取POST请求参数
GetPostFormArray(key string) ([]string, bool)
PostFormArray(key string) []string
GetPostForm(key string) (string, bool)
PostForm(key string) string
DefaultPostForm(key, defaultValue string) string

// data binding
Bind (obj interface {}) error // bind data according to Content-Type
BindJSON(obj interface{}) error
BindQuery(obj interface{}) error

ShouldBind(obj interface{}) error
ShouldBindJSON(obj interface{}) error
ShouldBindQuery(obj interface{}) error
...

其中Bind相关的函数是一种比较通用的方法,它允许我们将请求参数填充到一个map或struct中,这样在后续请求处理时,能够方便的使用传入的参数。Bind相关的函数分为三大类:

// 1. Bind函数
Bind (obj interface {}) error // 根据请求中content-type的类型来选择对应的具体Bind函数,底层调用的是BindJson, BindQuery这种具体的Bind函数

// 2. BindXXX(),具体的Bind函数,用于绑定一种参数类型,底层调用MustBindWith或者ShouldBindWith
BindJSON(obj interface{}) error
BindQuery(obj interface{}) error

// 3. 最底层的基础函数
MustBindWith(obj any, b binding.Binding) error // 当出现参数校验问题时,会直接返回400,底层仍然是ShouldBindWith
ShouldBindWith(obj any, b binding.Binding) error
其中各种类型的Bind函数,最底层的调用都是ShouldBindWith()函数,该函数有两个参数,第一个参数obj为需要将参数填充进去的对象(本文称为填充对象);第二个参数为binding.Binding类型,该类型的定义在package binding中,如下:

type Binding interface {
Name() string
Bind(*http.Request, any) error
}
Bingding为一个接口,提供了Name()和Bind()两个函数,Name()负责返回对应的Bind类型,比如JSON,XML等,Bind()函数则负责实现具体类型的参数绑定。为了完成常用参数类型的绑定,gin给每种参数类型都定义了一个类,并实现Binding接口,具体实现了Binding接口的类如下:

// 实现Binding接口的具体类
var (
JSON = jsonBinding{}
XML = xmlBinding{}
Form = formBinding{}
Query = queryBinding{}
FormPost = formPostBinding{}
FormMultipart = formMultipartBinding{}
ProtoBuf = protobufBinding{}
MsgPack = msgpackBinding{}
YAML = yamlBinding{}
Uri = uriBinding{}
Header = headerBinding{}
TOML = tomlBinding{}
)
这样,每当一个HTTP请求到来的时候,用户可以直接调用Bind()函数,Bind()函数可以通过content-type选择对应的实现了Bind()方法的对应类的实例,调用其Bind()方法,完成参数绑定.

常用的HTTP请求参数类型为HTTP GET query参数类型和HTTP POST json参数类型,以这两个为例,看一下参数绑定的一些细节:

BindJSON:

jsonBinding对Binding的实现如下:

   json的参数绑定比较简单,使用json.Decoder完成

func (jsonBinding) Name() string {
return "json"
}

func (jsonBinding) Bind(req *http.Request, obj any) error {
if req == nil || req.Body == nil {
return errors.New("invalid request")
}
return decodeJSON(req.Body, obj)
}

func decodeJSON(r io.Reader, obj any) error {
decoder := json.NewDecoder(r) // 使用json.Decoder对json进行解析
if EnableDecoderUseNumber {
decoder.UseNumber()
}
if EnableDecoderDisallowUnknownFields {
decoder.DisallowUnknownFields()
}
if err := decoder.Decode(obj); err != nil {
return err
}
return validate(obj) // 参数校验
}

BindQuery:

一个HTTP GET请求的query参数,在go中,可以通过request.URL.Query()获取到,获取到的query参数类型为url.Values,因此,BindQuery()首先获取到url.Values类型的query参数,然后设置对应的值,BindQuery()根据传进来的要将参数填充进去的对象类型(本文称为填充对象,是map类型还是struct ptr),分成了两个填充函数:

/* mapFormByTag是queryBinding.Bind()的底层核心函数

  • ptr: 填充对象,可能是map或strcut的指针

  • form: url.Values,包含所有query参数

  • tag: 值为"form"
    */
    func mapFormByTag(ptr any, form map[string][]string, tag string) error {
    // Check if ptr is a map
    ptrVal := reflect.ValueOf(ptr)
    var pointed any
    if ptrVal.Kind() == reflect.Ptr {
    ptrVal = ptrVal.Elem()
    pointed = ptrVal.Interface()
    }
    if ptrVal.Kind() == reflect.Map &&
    ptrVal.Type().Key().Kind() == reflect.String {
    if pointed != nil {
    ptr = pointed
    }
    return setFormMap(ptr, form) // 如果填充对象是map类型
    }

    return mappingByPtr(ptr, formSource(form), tag) // 填充对象是ptr struct类型
    }

接下来分别看一下两个核心的填充函数setFormMap()和mappingByPtr()

// setFormMap本身比较简单,因为query参数(url.Values是map[string][]string类型的别称)本身就是map[string][]string类型,只需要判断填充对象是map[string]string还是map[string][]string类型
func setFormMap(ptr any, form map[string][]string) error {
el := reflect.TypeOf(ptr).Elem() // 因为ptr本身是map类型,Elem返回该map的value值
// 如果map填充对象的value值为[]string类型,直接填充
if el.Kind() == reflect.Slice {
ptrMap, ok := ptr.(map[string][]string)
if !ok {
return ErrConvertMapStringSlice
}
for k, v := range form {
ptrMap[k] = v
}

    return nil
}
// 否则,map填充对象为map[string]string类型,取url.Values每个key对应的value值(类型为[]string)的最后一个元素填充到填充对象中
ptrMap, ok := ptr.(map[string]string)
if !ok {
    return ErrConvertToMapString
}
for k, v := range form {
    ptrMap[k] = v[len(v)-1] // pick last
}

return nil

}

mappingByPtr()的逻辑比较复杂,核心是遍历struct的每一个字段,获取该struct字段的json tag,如果tag的值跟某一个url.Values的key的值相等,填充对应的值

func mappingByPtr(ptr any, setter setter, tag string) error {
_, err := mapping(reflect.ValueOf(ptr), emptyField, setter, tag)
return err
}

/* mapping是一个递归函数,因为struct填充对象有可能嵌套了ptr成员

  • value:填充对象

  • field:struct填充对象的某个具体成员变量

  • setter:内部包含了url.Values

  • tag: 等于"form"
    */
    func mapping(value reflect.Value, field reflect.StructField, setter setter, tag string) (bool, error) {
    if field.Tag.Get(tag) == "-" { // just ignoring this field
    return false, nil
    }

    vKind := value.Kind()

    // 如果填充对象是ptr,获取其指向的对象递归调用mapping
    if vKind == reflect.Ptr {
    var isNew bool
    vPtr := value
    if value.IsNil() {
    isNew = true
    vPtr = reflect.New(value.Type().Elem())
    }
    isSet, err := mapping(vPtr.Elem(), field, setter, tag)
    if err != nil {
    return false, err
    }
    if isNew && isSet {
    value.Set(vPtr)
    }
    return isSet, nil
    }
    // 递归到最底层,每个value都是StructField类型,开始填充值
    if vKind != reflect.Struct || !field.Anonymous {
    ok, err := tryToSetValue(value, field, setter, tag)
    if err != nil {
    return false, err
    }
    if ok {
    return true, nil
    }
    }
    // 如果填充对象是struct,针对每一个struct field,递归调用mapping
    if vKind == reflect.Struct {
    tValue := value.Type()

      var isSet bool
      for i := 0; i < value.NumField(); i++ {
          sf := tValue.Field(i)
          if sf.PkgPath != "" && !sf.Anonymous { // unexported
              continue
          }
          ok, err := mapping(value.Field(i), sf, setter, tag)
          if err != nil {
              return false, err
          }
          isSet = isSet || ok
      }
      return isSet, nil
    

    }
    return false, nil
    }

总结: Gin通过区分不同的参数类型,每种参数类型实现了统一的Bind()函数来完成对应的参数绑定,用户只需要调用统一的函数,不用关系底层实现细节,即可完成参数绑定

标签:map,return,string,Bind,学习,参数,context,gin,函数
From: https://www.cnblogs.com/oldking1002/p/18569454

相关文章

  • 11 Nginx搭建(2)
    一、安装#安装nginx,要支持http2需要nginx在1.9.5以上、openssl在1.0.2及以上#如果openssl版本在1.0.2以下,需要安装源码包openssl、并在nginx的configure中写--with-openssl选项#下载openssl:https://www.openssl.org/source/-----------------------#查看系统的openssl......
  • 鸿蒙学习全方位运维分析
    文章目录1、崩溃服务2、性能管理3、云服务监控4、故障监控和预防HUAWEIAppGalleryConnect提供低门槛、高效率、多场景的大数据能力,包括质量分析、性能调优、故障定位、行业风向等。同时支持多维度数据分析,智能诊断问题并给出解决方案,为开发者明确质量优化方向,提升......
  • 工作学习笔记(十五)Mybatis-Plus项目中使用eq
    在今天的工作中遇到了一个问题,在这记录一下第一次使用eq()。方法作用它的主要作用是在构建SQL查询语句的条件部分时,添加一个等于的判断条件。例如,当你想从数据库表中查询出某一字段值等于特定值的记录时,就可以使用eq()方法来实现这个条件构建。方法语法及参数说明语法:......
  • Java编程学习五
    一、数组的缺陷:二、集合框架三、Vector类四、ArrayList集合五、LinkedList集合六、泛型七、HashSet八、HashMap一、数组的缺陷:1.数组存在定容问题,一旦定义长度,就固定了容量,有时候定义的数据量不一定,很难保证容量不越出;如果需要存储更多或更少的元素,可能需要创建一个新......
  • Java编程学习六
    javaIO操作:Java中,Io操作主要是指使用Java进行输入和输出操作。Java中所有的IO机制都是基于数据流进行输入输出的。这些数据流表示了字符或者字节数据的流动序列。在Java中进行io操作,通常需要用到Java.io包中的类。比如说fiel类,用于表示文件和目录路径名的抽象表示形式。。......
  • Java学习笔记——2024.11.25
    2024.11.25一、Java_DOS原理1.DOS基本原理创建文件夹=>mdd:\\xxx消除文件夹=>rdd:\\xxx2.相对路径和绝对路径=>相对路径:从当前目录开始定位,形成的一个路径=>绝对路径:从顶级目录d,开始定位,形成的路径举例子:相对路径:..\..\abc2\test200\hello.txt......
  • python爬虫学习之--抓取汽车之家数据
    汽车之家的数据爬取还是比较简单的,遇到的坑如下:页面的页面编码格式:汽车之家的页面编码格式有三种,分别是**“GB2312”,“ISO-8859-1"和"UTF-8-SIG”,每次使用requests模块获取页面的html时,会随机出现其中的一种,其中页面编码格式为"GB2312",“ISO-8859-1”,可以正常显......
  • 【linux学习指南】初识Linux进程信号与使用
    文章目录......
  • 学习笔记(四十六):$$语法:内置组件双向同步
    概述:$$运算符为系统内置组件提供TS变量的引用,使得TS变量和系统内置组件的内部状态保持同步使用规则:1、当前$$支持基础类型变量,以及@State、@Link和@Prop装饰的变量2、$$绑定的变量变化时,会触发UI的同步刷新3、支持的组件 使用示例:@Entity@ComponentexportstructLog......
  • Java学习-9
    一、字符串转字符在Java中,可以通过 String 类的 charAt(intindex) 方法将字符串转换为字符。这个方法返回指定索引位置的字符,索引从0开始。示例代码:importjava.util.Scanner;publicclassStringToCharExample{publicstaticvoidmain(String[]args){Scann......