gin.context学习

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.handlers[c.index](c)

    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()函数来完成对应的参数绑定,用户只需要调用统一的函数,不用关系底层实现细节,即可完成参数绑定

posted @   随心行者  阅读(62)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
点击右上角即可分享
微信分享提示