先直接上代码

r := gin.Default()
	// 注册中间件,使body可以重复读取
	r.Use(func(context *gin.Context) {
		all, err := context.GetRawData() // 读取body的内容
		if err != nil {
			log.Fatal(err)
		}
		// 重写 GetBody 方法,以便后续的其他操作
		context.Request.GetBody = func() (io.ReadCloser, error) {
			context.Request.Body = io.NopCloser(bytes.NewBuffer(all))
			buffer := bytes.NewBuffer(all)
			closer := io.NopCloser(buffer)
			return closer, nil
		}
		body, _ := context.Request.GetBody()  // 每次调用GetBody方法,都会新生成一个io.ReadCloser,但是底层的byte数据,都是all变量缓存的。
		context.Request.Body = body
		context.Next()
	})

注意,上面的中间件,需要在第一个执行。

分析

在gin中,context.Request.Body 是一个io.ReadCloser的接口,如下图

gin自定义中间件解决requestBody不可重复读问题(最新推荐)

查看io.ReadCloser接口定义

type ReadCloser interface {
	Reader
	Closer
}
type Reader interface {
	Read(p []byte) (n int, err error)
}
type Closer interface {
	Close() error
}

我们发现io.ReaderCloser接口的本质就是Read(p []byte) (n int, err error) 和 Close() error 的组合。

所以我们只需要自己编写实现Read(p []byte) (n int, err error) 和 Close() error 这两个方法的结构体即可赋值给context.Request.Body,在我们自己实现的方法中实现可重复读取即可达到我们的目的。

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。