golang gin 框架读取无法用 body 传递的表单参数

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: golang gin 框架读取无法用 body 传递的表单参数

这两天发现一个有意思的情况,好像 gin 框架里 c.PostForm() 函数只能从表单中获取参数,不能从 body 中解析表单参数,也就是说你如果用 c.PostForm() 来解析获取参数,客户端发起请求时,如果参数放在表单里,服务端能正常获取到参数,但是如果客户端把参数放在 body 里,即使 header 里配置了 content-type:multipart/form-data,服务端仍就无法获取到参数。下面是检验过程:

服务端代码

package main
import (
  "bytes"
  "fmt"
  "github.com/gin-gonic/gin"
  "io/ioutil"
)
func main() {
  router := gin.Default()
  router.POST("/test", func(c *gin.Context) {
    // 打印出 body
    data, _ := ioutil.ReadAll(c.Request.Body)
    fmt.Printf("req.body=%s\n, content-type=%v\n", data, c.ContentType())
    // 把字节流重新放回 body 中
    c.Request.Body = ioutil.NopCloser(bytes.NewBuffer(data))
    // 获取参数
    userName := c.PostForm("user_name")
    age := c.PostForm("age")
    fmt.Printf("userName=%s, age=%s\n", userName, age)
    c.JSON(200, "success")
  })
  router.Run(":8080")
}

postman 发起请求

使用 表单传递参数

1456655-20220807155144346-391630701.png

查看服务端控制台,打印出了 form-data 格式的 body,也打印出了userName 和 age 参数,通过输出内容可以看出请求体的 content-type 是 multipart/form-data

1456655-20220807155529786-1244078553.png

使用 body 传参,并配合 header 里配置 content-type:multipart/form-data

1456655-20220807155747198-2134696429.png1456655-20220807155915947-932927767.png

查看服务端控制台,发现 body内容这次不再是 form-data 格式,而是 json字符串的形式,即使打印出的 content-type 为 multipart/form-data 表明了我们所传递的参数类型确实是表单类型,但是参数还是没有获取成功

1456655-20220807160149608-1695768005.png

仔细看其实控制台有一条 error 信息error on parse multipart form array: no multipart boundary param in Content-Type, 中文意思是转换表单数组时发生错误,在当前 content-type 下,没有可以转换的表单参数。说明服务端确实识别到了 content-type 是表单类型,但是在对 body 按照表单形式获取参数时,解析失败了。这样就验证了 c.PostForm() 函数只能从表单中获取参数,不能从 body 中解析表单参数。

by the way

1、c.Request.Body 是一个 io.CloseReader 类型,并不是字节数组类型或者字符串类型,所以如果想获取到真实 body 内容,还需要通过 ioutil.ReadAll() 来获取。

2、通过 ioutil.ReadAll() 来读取完 body 内容后,body 就为空了,为了后面能继续从 body 中获取参数,需要通过 c.Request.Body = ioutil.NopCloser(bytes.NewBuffer(data)) 将字节数组重新返回到 body 中。

data, _ := ioutil.ReadAll(c.Request.Body)
fmt.Printf("req.body=%s\n, content-type=%v\n", data, c.ContentType())
// 把字节流重新放回 body 中
c.Request.Body = ioutil.NopCloser(bytes.NewBuffer(data))

参考

Gin框架body参数获取

Gin中间件打印request.Body

http的请求体body的几种数据格式https://blog.csdn.net/qq_41063141/article/details/101505956



相关文章
|
6月前
|
负载均衡 监控 Go
使用Golang框架构建分布式系统
本文探讨了使用Golang构建分布式系统的方法。Golang因其高效、简洁的语法和并发支持成为理想的开发语言。文中列举了几个常用的Golang框架,如Echo、Gin、gRPC和NATS等,并强调了服务拆分、通信机制、负载均衡等构建分布式系统的关键要素。通过选择合适的框架,遵循需求分析、技术选型、服务设计等步骤,开发者可以构建出高性能、高可用和可扩展的系统。此外,文中还提供了一个使用gRPC和etcd的简单代码案例来说明实现过程。
324 4
|
4月前
|
关系型数据库 API Go
[golang]在Gin框架中使用JWT鉴权
[golang]在Gin框架中使用JWT鉴权
129 0
|
4月前
|
网络协议 Go
[golang]gin框架接收websocket通信
[golang]gin框架接收websocket通信
126 0
|
7月前
|
分布式计算 Java Go
Golang深入浅出之-Go语言中的分布式计算框架Apache Beam
【5月更文挑战第6天】Apache Beam是一个统一的编程模型,适用于批处理和流处理,主要支持Java和Python,但也提供实验性的Go SDK。Go SDK的基本概念包括`PTransform`、`PCollection`和`Pipeline`。在使用中,需注意类型转换、窗口和触发器配置、资源管理和错误处理。尽管Go SDK文档有限,生态系统尚不成熟,且性能可能不高,但它仍为分布式计算提供了可移植的解决方案。通过理解和掌握Beam模型,开发者能编写高效的数据处理程序。
253 1
|
7月前
|
Go
golang学习3,golang 项目中配置gin的web框架
golang学习3,golang 项目中配置gin的web框架
|
3月前
|
Go
Golang语言之管道channel快速入门篇
这篇文章是关于Go语言中管道(channel)的快速入门教程,涵盖了管道的基本使用、有缓冲和无缓冲管道的区别、管道的关闭、遍历、协程和管道的协同工作、单向通道的使用以及select多路复用的详细案例和解释。
144 4
Golang语言之管道channel快速入门篇
|
3月前
|
Go
Golang语言文件操作快速入门篇
这篇文章是关于Go语言文件操作快速入门的教程,涵盖了文件的读取、写入、复制操作以及使用标准库中的ioutil、bufio、os等包进行文件操作的详细案例。
72 4
Golang语言文件操作快速入门篇
|
3月前
|
Go
Golang语言之gRPC程序设计示例
这篇文章是关于Golang语言使用gRPC进行程序设计的详细教程,涵盖了RPC协议的介绍、gRPC环境的搭建、Protocol Buffers的使用、gRPC服务的编写和通信示例。
119 3
Golang语言之gRPC程序设计示例
|
3月前
|
安全 Go
Golang语言goroutine协程并发安全及锁机制
这篇文章是关于Go语言中多协程操作同一数据问题、互斥锁Mutex和读写互斥锁RWMutex的详细介绍及使用案例,涵盖了如何使用这些同步原语来解决并发访问共享资源时的数据安全问题。
101 4
|
3月前
|
Go 调度
Golang语言goroutine协程篇
这篇文章是关于Go语言goroutine协程的详细教程,涵盖了并发编程的常见术语、goroutine的创建和调度、使用sync.WaitGroup控制协程退出以及如何通过GOMAXPROCS设置程序并发时占用的CPU逻辑核心数。
79 4
Golang语言goroutine协程篇