HTTP请求中的form data和request payload的区别(request 后台无法获取参数)

Stella981
• 阅读 930

转载自:btg.yoyo

 jQuery的ajax方法和post方法分别发送请求,在后台Servlet进行处理时结果是不一样的,比如用$.ajax方法发送请求时(data参数是一个JSON.stringify()处理后的字符串,而不是一个JSON对象),servlet里可以这样使用Gson来解析:

new Jsonparser().parse(request.getReader())

但此时是不可用request.getParam(key) 来取值的。

如果用$.post方法来发送请求(data参数是一个JSON对象,而不要再用JSON.stringify()处理为字符串了),结果恰恰相反。

在Chrome中调试发现,$.ajax发送的请求显示在request payload下面,而使用$.post方法发送的请求显示在form data下面。有什么区别呢?

关键就是设置Content-type这个Header为application/x-www-form-urlencoded,实际上对于常规的HTML页面上的form的Content-type默认就是这个值。

这里要注意post请求的Content-Type为application/x-www-form-urlencoded,参数是在请求体中,即上面请求中的Form Data。

在servlet中,可以通过request.getParameter(name)的形式来获取表单参数。

 而如果使用原生AJAX POST请求的话:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

function  getXMLHttpRequest() {

var  xhr;

if (window.ActiveXObject) {

xhr=  new  ActiveXObject( "Microsoft.XMLHTTP" );

} else  if  (window.XMLHttpRequest) {

xhr=  new  XMLHttpRequest();

} else  {

xhr=  null ;

}

return  xhr;

}

function  save() {

var  xhr = getXMLHttpRequest();

xhr.open( "post" , "http://127.0.0.1:8080/test/test.do" );

var  data =  "name=mikan&address=street..." ;

xhr.send(data);

xhr.onreadystatechange=  function () {

if (xhr.readyState == 4 && xhr.status == 200) {

alert( "returned:" + xhr.responseText);

}

};

}

  通过chrome的开发者工具看到请求头如下

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

RequestURL:http: //127.0.0.1:8080/test/test.do

Request Method:POST

Status Code:200 OK

Request Headers

Accept:*/*

Accept-Encoding:gzip,deflate,sdch

Accept-Language:zh-CN,zh;q=0.8,en;q=0.6

AlexaToolbar-ALX_NS_PH:AlexaToolbar/alxg-3.2

Connection:keep-alive

Content-Length:28

Content-Type:text/plain;charset=UTF-8

Cookie:JSESSIONID=C40C7823648E952E7C6F7D2E687A0A89

Host:127.0.0.1:8080

Origin:http: //127.0.0.1:8080

Referer:http: //127.0.0.1:8080/test/index.jsp

User-Agent:Mozilla/5.0 (Windows NT 6.1)AppleWebKit/537.36 (KHTML, like Gecko) Chrome/33.0.1750.149 Safari/537.36

Request Payload

name=mikan&address=street

Response Headers

Content-Length:2

Date:Sun, 11 May 2014 11:49:23 GMT

Server:Apache-Coyote/1.1

  注意请求的Content-Type为text/plain;charset=UTF-8,而请求表单参数在RequestPayload中。

那么servlet中通过request.getParameter(name)却是空。为什么呢?而这样的参数又该怎么样获取呢?

为了搞明白这个问题,查了些资料,也看了Tomcat7.0.53关于请求参数处理的源码,终于搞明白了是怎么回事。

HTTP POST表单请求提交时,使用的Content-Type是application/x-www-form-urlencoded,而使用原生AJAX的POST请求如果不指定请求头RequestHeader,默认使用的Content-Type是text/plain;charset=UTF-8。

 由于Tomcat对于Content-Type multipart/form-data(文件上传)和application/x-www-form-urlencoded(POST请求)做了“特殊处理”。下面来看看相关的处理代码。

Tomcat的HttpServletRequest类的实现类为org.apache.catalina.connector.Request(实际上是org.apache.coyote.Request),而它对处理请求参数的方法为protected void parseParameters(),这个方法中对Content-Type multipart/form-data(文件上传)和application/x-www-form-urlencoded(POST请求)的处理代码如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

protectedvoid parseParameters() {

//省略部分代码......

parameters.handleQueryParameters(); // 这里是处理url中的参数

//省略部分代码......

if  ( "multipart/form-data" .equals(contentType)) {  // 这里是处理文件上传请求

parseParts();

success =  true ;

return ;

}

if (!( "application/x-www-form-urlencoded" .equals(contentType))) { // 这里如果是非POST请求直接返回,不再进行处理

success =  true ;

return ;

}

//下面的代码才是处理POST请求参数

//省略部分代码......

try  {

if  (readPostBody(formData, len)!= len) {  // 读取请求体数据

return ;

}

catch  (IOException e) {

// Client disconnect

if (context.getLogger().isDebugEnabled()) {

context.getLogger().debug(

sm.getString( "coyoteRequest.parseParameters" ),e);

}

return ;

}

parameters.processParameters(formData,  0 , len);  // 处理POST请求参数,把它放到requestparameter map中(即request.getParameterMap获取到的Map,request.getParameter(name)也是从这个Map中获取的)

// 省略部分代码......

}

protected  int  readPostBody( byte  body[],  int  len)

throws  IOException {

int  offset =  0 ;

do  {

int  inputLen = getStream().read(body, offset, len - offset);

if  (inputLen <=  0 ) {

return  offset;

}

offset += inputLen;

while  ((len - offset) >  0 );

return  len;

}

  从上面代码可以看出,Content-Type不是application/x-www-form-urlencoded的POST请求是不会读取请求体数据和进行相应的参数处理的,即不会解析表单数据来放到request parameter map中。所以通过request.getParameter(name)是获取不到的。

 那么这样提交的参数我们该怎么获取呢?

当然是使用最原始的方式,读取输入流来获取了,如下所示:

1

2

3

4

5

6

7

8

9

10

11

12

13

privateString getRequestPayload(HttpServletRequest req) {

StringBuildersb =  new  StringBuilder();

try (BufferedReaderreader = req.getReader();) {

char []buff =  new  char [ 1024 ];

intlen;

while ((len = reader.read(buff)) != - 1 ) {

sb.append(buff, 0 , len);

}

} catch  (IOException e) {

e.printStackTrace();

}

returnsb.toString();

}

当然,设置了application/x-www-form-urlencoded的POST请求也可以通过这种方式来获取。

 所以,在使用原生AJAX POST请求时,需要明确设置Request Header,即:

 xhr.setRequestHeader("Content-Type","application/x-www-form-urlencoded");

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Wesley13 Wesley13
3年前
java将前端的json数组字符串转换为列表
记录下在前端通过ajax提交了一个json数组的字符串,在后端如何转换为列表。前端数据转化与请求varcontracts{id:'1',name:'yanggb合同1'},{id:'2',name:'yanggb合同2'},{id:'3',name:'yang
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
虾米大王 虾米大王
2年前
java代码012
code012.jspInserttitlehere<%JSP内置对象1.request用于处理HTTP请求中的各项参数。如,删除可以通过request对象的getParameter()方法获取如,request.getParameter("id")在请求转发时,需要把一些数据传递到转发后的页面处理。就需要用到request的se
Wesley13 Wesley13
3年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Stella981 Stella981
3年前
Google地球出现“无法连接到登录服务器(错误代码:c00a0194)”解决方法
Google地球出现“无法连接到登录服务器(错误代码:c00a0194)”解决方法参考文章:(1)Google地球出现“无法连接到登录服务器(错误代码:c00a0194)”解决方法(https://www.oschina.net/action/GoToLink?urlhttps%3A%2F%2Fwww.codeprj.com%2Fblo
Python进阶者 Python进阶者
10个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这