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

简介: 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



相关文章
|
1天前
|
SQL 关系型数据库 MySQL
探索Gorm - Golang流行的数据库ORM框架
探索Gorm - Golang流行的数据库ORM框架
|
1天前
|
Go
golang学习3,golang 项目中配置gin的web框架
golang学习3,golang 项目中配置gin的web框架
|
1天前
|
分布式计算 Java Go
Golang深入浅出之-Go语言中的分布式计算框架Apache Beam
【5月更文挑战第6天】Apache Beam是一个统一的编程模型,适用于批处理和流处理,主要支持Java和Python,但也提供实验性的Go SDK。Go SDK的基本概念包括`PTransform`、`PCollection`和`Pipeline`。在使用中,需注意类型转换、窗口和触发器配置、资源管理和错误处理。尽管Go SDK文档有限,生态系统尚不成熟,且性能可能不高,但它仍为分布式计算提供了可移植的解决方案。通过理解和掌握Beam模型,开发者能编写高效的数据处理程序。
140 1
|
1天前
|
XML JSON 人工智能
探索Gin框架:Golang Gin框架请求参数的获取
探索Gin框架:Golang Gin框架请求参数的获取
|
1天前
|
存储 人工智能 Go
探索Gin框架:Golang使用Gin完成文件上传
探索Gin框架:Golang使用Gin完成文件上传
|
1天前
|
存储 中间件 Go
探索Gin框架:快速构建高性能的Golang Web应用
探索Gin框架:快速构建高性能的Golang Web应用
|
1天前
|
监控 算法 Go
Golang深入浅出之-Go语言中的服务熔断、降级与限流策略
【5月更文挑战第4天】本文探讨了分布式系统中保障稳定性的重要策略:服务熔断、降级和限流。服务熔断通过快速失败和暂停故障服务调用来保护系统;服务降级在压力大时提供有限功能以保持整体可用性;限流控制访问频率,防止过载。文中列举了常见问题、解决方案,并提供了Go语言实现示例。合理应用这些策略能增强系统韧性和可用性。
50 0
|
1天前
|
缓存 测试技术 持续交付
Golang深入浅出之-Go语言中的持续集成与持续部署(CI/CD)
【5月更文挑战第5天】本文介绍了Go语言项目中的CI/CD实践,包括持续集成与持续部署的基础知识,常见问题及解决策略。测试覆盖不足、版本不一致和构建时间过长是主要问题,可通过全面测试、统一依赖管理和利用缓存优化。文中还提供了使用GitHub Actions进行自动化测试和部署的示例,强调了持续优化CI/CD流程以适应项目需求的重要性。
54 1
|
1天前
|
Kubernetes Cloud Native Go
Golang深入浅出之-Go语言中的云原生开发:Kubernetes与Docker
【5月更文挑战第5天】本文探讨了Go语言在云原生开发中的应用,特别是在Kubernetes和Docker中的使用。Docker利用Go语言的性能和跨平台能力编写Dockerfile和构建镜像。Kubernetes,主要由Go语言编写,提供了方便的客户端库与集群交互。文章列举了Dockerfile编写、Kubernetes资源定义和服务发现的常见问题及解决方案,并给出了Go语言构建Docker镜像和与Kubernetes交互的代码示例。通过掌握这些技巧,开发者能更高效地进行云原生应用开发。
54 1
|
1天前
|
负载均衡 监控 Go
Golang深入浅出之-Go语言中的服务网格(Service Mesh)原理与应用
【5月更文挑战第5天】服务网格是处理服务间通信的基础设施层,常由数据平面(代理,如Envoy)和控制平面(管理配置)组成。本文讨论了服务发现、负载均衡和追踪等常见问题及其解决方案,并展示了使用Go语言实现Envoy sidecar配置的例子,强调Go语言在构建服务网格中的优势。服务网格能提升微服务的管理和可观测性,正确应对问题能构建更健壮的分布式系统。
30 1

相关实验场景

更多