Get和Post各种编码方式和获取参数的问题

2024-02-07 15:38

本文主要是介绍Get和Post各种编码方式和获取参数的问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Get和Post各种编码方式和获取参数的问题

  做后端开发,会跟各种请求打交道,Get或者Post甚至其他,而就Get和Post不同的编码方式和content-type提交也有许多的不同,在开发过程中,经常遇到参数接收不到的问题,无论是原生servlet还是springboot框架,都有到过这种问题。这篇文中就这些问题,通过分析,希望能让自己能更进一步的了解HTTP请求和参数接收相关的原理。

Get请求

  Get请求比较简单,请求的参数为[key=value],通过&连接起来,一般被称为查询字符串(Query String),例如:http://127.0.0.1/test?name=strive&age=23,参数为:name=strive&age=23.

GET /get?name=strive HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.120 Safari/537.36
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3
Sec-Fetch-Site: same-origin
Referer: http://localhost:8080/getAndPostIndex
Accept-Encoding: gzip, deflate, br

  其实我们只需要看/get?name=strive即查询参数,GET请求参数是被存放在QueryString 中的,这种在我们后台是可以通过request.getParameter() 获取请求参数的。

POST请求

  下面我们来看POST请求,POST请求的方式比Get复杂,它参数的存放位置是和Content-Type 有关系的,具体有什么关系,继续往下看。

Content-Type为application/x-www-form-urlencoded
POST /postUrl HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Content-Length: 11
Cache-Control: max-age=0
Origin: http://localhost:8080
Upgrade-Insecure-Requests: 1
Content-Type: application/x-www-form-urlencoded
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.120 Safari/537.36
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3
Sec-Fetch-Site: same-origin
Referer: http://localhost:8080/getAndPostIndex
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9,ko;q=0.8,en;q=0.7
Form Data
name=strive

  这种方式提交的参数放在了以[key=value]方式存在了请求体中,这种在我们后台是也是可以通过request.getParameter() 获取请求参数的。

Content-Type为multipart/form-data
POST /postFileUpload HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Content-Length: 3768
Cache-Control: max-age=0
Origin: http://localhost:8080
Upgrade-Insecure-Requests: 1
#Content-Type为multipart/form-data
Content-Type: multipart/form-data; boundary=----WebKitFormBoundary6A2xLxBKaSvqBvDd
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.120 Safari/537.36
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3
Sec-Fetch-Site: same-origin
Referer: http://localhost:8080/getAndPostIndex
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9,ko;q=0.8,en;q=0.7
#Request的请求体
Request Payload
------WebKitFormBoundary6A2xLxBKaSvqBvDd
Content-Disposition: form-data; name="file"; filename="Baidu.cer"
Content-Type: application/x-x509-ca-cert
#这里省略了文件的数据
------WebKitFormBoundary6A2xLxBKaSvqBvDd
Content-Disposition: form-data; name="name"
strive
------WebKitFormBoundary6A2xLxBKaSvqBvDd--

  在Content-Type: multipart/form-data 时,而且POST提交的参数存放在了Request Payload,这种请求参数没办法通过request.getParameter() 获取请求参数的。

Content-Type为application/json
POST /postJson HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Content-Length: 13
Accept: */*
Origin: http://localhost:8080
X-Requested-With: XMLHttpRequest
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.120 Safari/537.36
Sec-Fetch-Mode: cors
Content-Type: application/json
Sec-Fetch-Site: same-origin
Referer: http://localhost:8080/getAndPostIndex
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9,ko;q=0.8,en;q=0.7
#Request的请求体
Request Payload
{name:strive}

  在Content-Type: application/json 时,而且POST提交的参数存放在了Request Payload,这种请求参数没办法通过request.getParameter() 获取请求参数的。

Content-Type为text/plain
POST /postPlain HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Content-Length: 13
Cache-Control: max-age=0
Origin: http://localhost:8080
Upgrade-Insecure-Requests: 1
Content-Type: text/plain
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.120 Safari/537.36
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3
Sec-Fetch-Site: same-origin
Referer: http://localhost:8080/getAndPostIndex
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9,ko;q=0.8,en;q=0.7
#Request的请求体
Request Payload
name=strive

  在Content-Type: text/plain 时,而且POST提交的参数存放在了Request Payload,这种请求参数没办法通过request.getParameter() 获取请求参数的。

POST请求的问题

  从上面的几个请求的报文的例子中我们可以找到一些规律,以application/x-www-form-urlencoded作POST提交的数据,后台是可以通过request.getParameter() 获取请求参数的。而multipart/form-data,application/jsontext/plain作POST提交数据时,后台没法通过request.getParameter() 获取请求参数的。application/x-www-form-urlencoded的参数放在了名为Form Data的请求体中,而其他的则放在了名为Request Payload的请求体中。那是什么原因了?
  解析请求参数的实现源码org.apache.catalina.connector.Request#parseParameters如下:

 protected void parseParameters() {this.parametersParsed = true;Parameters parameters = this.coyoteRequest.getParameters();boolean success = false;try {parameters.setLimit(this.getConnector().getMaxParameterCount());Charset charset = this.getCharset();boolean useBodyEncodingForURI = this.connector.getUseBodyEncodingForURI();parameters.setCharset(charset);if (useBodyEncodingForURI) {parameters.setQueryStringCharset(charset);}parameters.handleQueryParameters();if (this.usingInputStream || this.usingReader) {success = true;return;}//获取请求的contentTypeString contentType = this.getContentType();if (contentType == null) {contentType = "";}int semicolon = contentType.indexOf(59);if (semicolon >= 0) {contentType = contentType.substring(0, semicolon).trim();} else {contentType = contentType.trim();}//处理文件上传if ("multipart/form-data".equals(contentType)) {this.parseParts(false);success = true;return;}if (!this.getConnector().isParseBodyMethod(this.getMethod())) {success = true;return;}if ("application/x-www-form-urlencoded".equals(contentType)) {//如果是这种contenttype则解析int len = this.getContentLength();if (len <= 0) {if ("chunked".equalsIgnoreCase(this.coyoteRequest.getHeader("transfer-encoding"))) {Object var21 = null;Context context;byte[] formData;try {formData = this.readChunkedPostBody();} catch (IllegalStateException var17) {parameters.setParseFailedReason(FailReason.POST_TOO_LARGE);context = this.getContext();if (context != null && context.getLogger().isDebugEnabled()) {context.getLogger().debug(sm.getString("coyoteRequest.parseParameters"), var17);}return;} catch (IOException var18) {parameters.setParseFailedReason(FailReason.CLIENT_DISCONNECT);context = this.getContext();if (context != null && context.getLogger().isDebugEnabled()) {context.getLogger().debug(sm.getString("coyoteRequest.parseParameters"), var18);}return;}if (formData != null) {parameters.processParameters(formData, 0, formData.length);}}} else {int maxPostSize = this.connector.getMaxPostSize();Context context;if (maxPostSize >= 0 && len > maxPostSize) {context = this.getContext();if (context != null && context.getLogger().isDebugEnabled()) {context.getLogger().debug(sm.getString("coyoteRequest.postTooLarge"));}this.checkSwallowInput();parameters.setParseFailedReason(FailReason.POST_TOO_LARGE);return;}context = null;byte[] formData;if (len < 8192) {if (this.postData == null) {this.postData = new byte[8192];}formData = this.postData;} else {formData = new byte[len];}try {if (this.readPostBody(formData, len) != len) {parameters.setParseFailedReason(FailReason.REQUEST_BODY_INCOMPLETE);return;}} catch (IOException var19) {Context context = this.getContext();if (context != null && context.getLogger().isDebugEnabled()) {context.getLogger().debug(sm.getString("coyoteRequest.parseParameters"), var19);}parameters.setParseFailedReason(FailReason.CLIENT_DISCONNECT);return;}// 处理POST请求参数,把它放到requestparameter map中(即request.getParameterMap获取到的Map,request.getParameter(name)也是从这个Map中获取的)parameters.processParameters(formData, 0, len);}success = true;return;}success = true;} finally {if (!success) {parameters.setParseFailedReason(FailReason.UNKNOWN);}}}

这篇关于Get和Post各种编码方式和获取参数的问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/688164

相关文章

golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法

《golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法》:本文主要介绍golang获取当前时间、时间戳和时间字符串及它们之间的相互转换,本文通过实例代码给大家介绍的非常详细,感兴趣... 目录1、获取当前时间2、获取当前时间戳3、获取当前时间的字符串格式4、它们之间的相互转化上篇文章给大家介

如何解决mmcv无法安装或安装之后报错问题

《如何解决mmcv无法安装或安装之后报错问题》:本文主要介绍如何解决mmcv无法安装或安装之后报错问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录mmcv无法安装或安装之后报错问题1.当我们运行YOwww.chinasem.cnLO时遇到2.找到下图所示这里3.

浅谈配置MMCV环境,解决报错,版本不匹配问题

《浅谈配置MMCV环境,解决报错,版本不匹配问题》:本文主要介绍浅谈配置MMCV环境,解决报错,版本不匹配问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录配置MMCV环境,解决报错,版本不匹配错误示例正确示例总结配置MMCV环境,解决报错,版本不匹配在col

Vue3使用router,params传参为空问题

《Vue3使用router,params传参为空问题》:本文主要介绍Vue3使用router,params传参为空问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录vue3使用China编程router,params传参为空1.使用query方式传参2.使用 Histo

SpringBoot首笔交易慢问题排查与优化方案

《SpringBoot首笔交易慢问题排查与优化方案》在我们的微服务项目中,遇到这样的问题:应用启动后,第一笔交易响应耗时高达4、5秒,而后续请求均能在毫秒级完成,这不仅触发监控告警,也极大影响了用户体... 目录问题背景排查步骤1. 日志分析2. 性能工具定位优化方案:提前预热各种资源1. Flowable

Python获取中国节假日数据记录入JSON文件

《Python获取中国节假日数据记录入JSON文件》项目系统内置的日历应用为了提升用户体验,特别设置了在调休日期显示“休”的UI图标功能,那么问题是这些调休数据从哪里来呢?我尝试一种更为智能的方法:P... 目录节假日数据获取存入jsON文件节假日数据读取封装完整代码项目系统内置的日历应用为了提升用户体验,

微信公众号脚本-获取热搜自动新建草稿并发布文章

《微信公众号脚本-获取热搜自动新建草稿并发布文章》本来想写一个自动化发布微信公众号的小绿书的脚本,但是微信公众号官网没有小绿书的接口,那就写一个获取热搜微信普通文章的脚本吧,:本文主要介绍微信公众... 目录介绍思路前期准备环境要求获取接口token获取热搜获取热搜数据下载热搜图片给图片加上标题文字上传图片

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

SpringBoot启动报错的11个高频问题排查与解决终极指南

《SpringBoot启动报错的11个高频问题排查与解决终极指南》这篇文章主要为大家详细介绍了SpringBoot启动报错的11个高频问题的排查与解决,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一... 目录1. 依赖冲突:NoSuchMethodError 的终极解法2. Bean注入失败:No qu

MySQL新增字段后Java实体未更新的潜在问题与解决方案

《MySQL新增字段后Java实体未更新的潜在问题与解决方案》在Java+MySQL的开发中,我们通常使用ORM框架来映射数据库表与Java对象,但有时候,数据库表结构变更(如新增字段)后,开发人员可... 目录引言1. 问题背景:数据库与 Java 实体不同步1.1 常见场景1.2 示例代码2. 不同操作