vue axios POST请求中参数以form data和request payload形式的原因

HTTP请求中,如果是get请求,那么表单参数以name=value&name1=value1的形式附到url的后面,如果是post请求,那么表单参数是在请求体中,也是以name=value&name1=value1的形式在请求体中。通过chrome的开发者工具可以看到如下(这里是可读的形式,不是真正的HTTP请求协议的请求格式):

get请求:

[plain] view plain copy
  1. RequestURL:http://127.0.0.1:8080/test/test.do?name=mikan&address=street  
  2. Request Method:GET  
  3. Status Code:200 OK  
  4.    
  5. Request Headers  
  6. Accept:text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8  
  7. Accept-Encoding:gzip,deflate,sdch  
  8. Accept-Language:zh-CN,zh;q=0.8,en;q=0.6  
  9. AlexaToolbar-ALX_NS_PH:AlexaToolbar/alxg-3.2  
  10. Connection:keep-alive  
  11. Cookie:JSESSIONID=74AC93F9F572980B6FC10474CD8EDD8D  
  12. Host:127.0.0.1:8080  
  13. Referer:http://127.0.0.1:8080/test/index.jsp  
  14. User-Agent:Mozilla/5.0 (Windows NT 6.1)AppleWebKit/537.36 (KHTML, like Gecko) Chrome/33.0.1750.149 Safari/537.36  
  15.    
  16. Query String Parameters  
  17. name:mikan  
  18. address:street  
  19.    
  20. Response Headers  
  21. Content-Length:2  
  22. Date:Sun, 11 May 2014 10:42:38 GMT  
  23. Server:Apache-Coyote/1.1  

Post请求:

[plain] view plain copy
  1. RequestURL:http://127.0.0.1:8080/test/test.do  
  2. Request Method:POST  
  3. Status Code:200 OK  
  4.    
  5. Request Headers  
  6. Accept:text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8  
  7. Accept-Encoding:gzip,deflate,sdch  
  8. Accept-Language:zh-CN,zh;q=0.8,en;q=0.6  
  9. AlexaToolbar-ALX_NS_PH:AlexaToolbar/alxg-3.2  
  10. Cache-Control:max-age=0  
  11. Connection:keep-alive  
  12. Content-Length:25  
  13. Content-Type:application/x-www-form-urlencoded  
  14. Cookie:JSESSIONID=74AC93F9F572980B6FC10474CD8EDD8D  
  15. Host:127.0.0.1:8080  
  16. Origin:http://127.0.0.1:8080  
  17. Referer:http://127.0.0.1:8080/test/index.jsp  
  18. User-Agent:Mozilla/5.0 (Windows NT 6.1)AppleWebKit/537.36 (KHTML, like Gecko) Chrome/33.0.1750.149 Safari/537.36  
  19.    
  20. Form Data  
  21. name:mikan  
  22. address:street  
  23.    
  24. Response Headers  
  25. Content-Length:2  
  26. Date:Sun, 11 May 2014 11:05:33 GMT  
  27. Server:Apache-Coyote/1.1  

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

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

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

[javascript] view plain copy
  1. function getXMLHttpRequest() {  
  2.           var xhr;  
  3.           if(window.ActiveXObject) {  
  4.                    xhr= new ActiveXObject("Microsoft.XMLHTTP");  
  5.           }else if (window.XMLHttpRequest) {  
  6.                    xhr= new XMLHttpRequest();  
  7.           }else {  
  8.                    xhr= null;  
  9.           }  
  10.           return xhr;  
  11. }  
  12.   
  13. function save() {  
  14.           var xhr = getXMLHttpRequest();  
  15.           xhr.open("post","http://127.0.0.1:8080/test/test.do");  
  16.           var data = "name=mikan&address=street...";  
  17.           xhr.send(data);  
  18.           xhr.onreadystatechange= function() {  
  19.                    if(xhr.readyState == 4 && xhr.status == 200) {  
  20.                             alert("returned:"+ xhr.responseText);  
  21.                    }  
  22.           };  
  23. }  

 

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

[plain] view plain copy
  1. RequestURL:http://127.0.0.1:8080/test/test.do  
  2. Request Method:POST  
  3. Status Code:200 OK  
  4.    
  5. Request Headers  
  6. Accept:*/*  
  7. Accept-Encoding:gzip,deflate,sdch  
  8. Accept-Language:zh-CN,zh;q=0.8,en;q=0.6  
  9. AlexaToolbar-ALX_NS_PH:AlexaToolbar/alxg-3.2  
  10. Connection:keep-alive  
  11. Content-Length:28  
  12. Content-Type:text/plain;charset=UTF-8  
  13. Cookie:JSESSIONID=C40C7823648E952E7C6F7D2E687A0A89  
  14. Host:127.0.0.1:8080  
  15. Origin:http://127.0.0.1:8080  
  16. Referer:http://127.0.0.1:8080/test/index.jsp  
  17. User-Agent:Mozilla/5.0 (Windows NT 6.1)AppleWebKit/537.36 (KHTML, like Gecko) Chrome/33.0.1750.149 Safari/537.36  
  18.    
  19. Request Payload  
  20. name=mikan&address=street  
  21.    
  22. Response Headers  
  23. Content-Length:2  
  24. Date:Sun, 11 May 2014 11:49:23 GMT  
  25. 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请求)的处理代码如下:

[java] view plain copy
  1. protectedvoid parseParameters() {  
  2.            //省略部分代码......  
  3.            parameters.handleQueryParameters();// 这里是处理url中的参数  
  4.            //省略部分代码......  
  5.            if ("multipart/form-data".equals(contentType)) { // 这里是处理文件上传请求  
  6.                 parseParts();  
  7.                 success = true;  
  8.                 return;  
  9.            }  
  10.    
  11.            if(!("application/x-www-form-urlencoded".equals(contentType))) {// 这里如果是非POST请求直接返回,不再进行处理  
  12.                 success = true;  
  13.                 return;  
  14.            }  
  15.            //下面的代码才是处理POST请求参数  
  16.            //省略部分代码......  
  17.            try {  
  18.                 if (readPostBody(formData, len)!= len) { // 读取请求体数据  
  19.                     return;  
  20.                 }  
  21.            } catch (IOException e) {  
  22.                 // Client disconnect  
  23.                 if(context.getLogger().isDebugEnabled()) {  
  24.                     context.getLogger().debug(  
  25.                             sm.getString("coyoteRequest.parseParameters"),e);  
  26.                 }  
  27.                 return;  
  28.            }  
  29.            parameters.processParameters(formData, 0, len); // 处理POST请求参数,把它放到requestparameter map中(即request.getParameterMap获取到的Map,request.getParameter(name)也是从这个Map中获取的)  
  30.            // 省略部分代码......  
  31. }  
  32.    
  33.    protected int readPostBody(byte body[], int len)  
  34.        throws IOException {  
  35.    
  36.        int offset = 0;  
  37.        do {  
  38.            int inputLen = getStream().read(body, offset, len - offset);  
  39.            if (inputLen <= 0) {  
  40.                 return offset;  
  41.            }  
  42.            offset += inputLen;  
  43.        } while ((len - offset) > 0);  
  44.        return len;  
  45.     }  

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

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

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

[java] view plain copy
  1. privateString getRequestPayload(HttpServletRequest req) {  
  2.           StringBuildersb = new StringBuilder();  
  3.           try(BufferedReaderreader = req.getReader();) {  
  4.                    char[]buff = new char[1024];  
  5.                    intlen;  
  6.                    while((len = reader.read(buff)) != -1) {  
  7.                             sb.append(buff,0, len);  
  8.                    }  
  9.           }catch (IOException e) {  
  10.                    e.printStackTrace();  
  11.           }  
  12.           returnsb.toString();  
  13. }  

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

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

[javascript] view plain copy
  1. xhr.setRequestHeader("Content-Type","application/x-www-form-urlencoded");  

另外,如果使用jquery,我使用1.11.0这个版本来测试,$.ajax post请求是不需要明确设置这个请求头的,其他版本的本人没有亲自测试过。相信在1.11.0之后的版本也是不需要设置的。不过之前有的就不一定了。这个没有测试过。


2015-04-17后记:

最近在看书时才真正搞明白,服务器为什么会对表单提交和文件上传做特殊处理,因为表单提交数据是名值对的方式,且Content-Type为application/x-www-form-urlencoded,而文件上传服务器需要特殊处理,普通的post请求(Content-Type不是application/x-www-form-urlencoded)数据格式不固定,不一定是名值对的方式,所以服务器无法知道具体的处理方式,所以只能通过获取原始数据流的方式来进行解析。

jquery在执行post请求时,会设置Content-Type为application/x-www-form-urlencoded,所以服务器能够正确解析,而使用原生ajax请求时,如果不显示的设置Content-Type,那么默认是text/plain,这时服务器就不知道怎么解析数据了,所以才只能通过获取原始数据流的方式来进行解析请求数据。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/247646.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue-resource使用

vue-resource是一个http请求插件&#xff0c;遵循promise&#xff0c;类似jquery中ajax操作。 vue-resource已不被官方推荐&#xff0c;官方推荐axios插件来操作http协议。 vue-resource中提供的方法 get(url, [options]) head(url, [options]) delete(url, [options]) jso…

HttpHttps

http协议与https Http 客户端发送一个HTTP请求到服务器的请求消息包括以下格式&#xff1a; **请求行&#xff08;request line&#xff09;、请求头部&#xff08;header&#xff09;、空行 和请求数据四个部分组成。** Get请求例子&#xff0c;使用Charles抓取的request&…

vue2使用axios post跳坑,封装成模块

终于将vue-resource替换成axios了&#xff0c;其中像application/x-www-form-urlencoded发送的头信息以及返回的response结果这两点都需要注意一下。 其实https://github.com/mzabriskie/axios也有说明的。因为我在vue-resource中使用了Vue.http.options.emulateJSON true;&am…

axios使用

axios和vue-resource一样&#xff0c;是一个vue中操作http的插件&#xff0c;遵循promise&#xff0c;vue官方也推荐使用axios。 安装axios npm i axios -S axios也是在运行时需要的&#xff0c;所以要保存在dependencies中。 引入axios import axios from axios Vue.proto…

jQuery length 和 size()区别

jQuery length和size()区别总结如下&#xff1a; 1.length是属性&#xff0c;size()是方法。 2.如果你只是想获取元素的个数&#xff0c;两者效果一样既 $("img").length 和 $("img").size() 获取的值是一样的&#xff1b;但是如果是获取字符串的长…

一些关于自己的未来的东西

2019.7.4 自己大一建立对编程的基础认识&#xff0c;确实培养了一些兴趣&#xff0c;入了个门&#xff0c;不过没有接触到本质。大二加入到了学校的网站开发团队&#xff0c;对网站开发后端进行了学习&#xff0c;对后台开发也有了基础的学习吧&#xff0c;哈哈可能以后就是要走…

Javascript面向对象编程:构造函数的继承

今天要介绍的是&#xff0c;对象之间的"继承"的五种方法。 比如&#xff0c;现在有一个"动物"对象的构造函数。 function Animal(){ this.species "动物"; } 还有一个"猫"对象的构造函数。 function Cat(name,color){ this.name nam…

并发与多线程

并发 并发&#xff08;concurrency&#xff09;是指CPU在某个时间段内交替处理多任务的能力。每个CPU不可能只顾着执行某个进程&#xff0c;而让其他进程一直等待被执行。所以&#xff0c;CPU把可执行时间均分成若干份&#xff0c;每个进程执行一份或多份时间后&#xff0c;记录…

有没有朋友可以帮我解释一下贴水是什么意思?

通俗易懂的讲&#xff1a;贴水便宜&#xff0c;升水贵 当前&#xff0c;螺纹钢05合约就是贴水01合约 翻译&#xff0c;螺纹钢05合约就是比01合约便宜 升水同理 转载于:https://www.cnblogs.com/luoluo-123/p/11142229.html

es6常用点记录

letconst解构赋值字符串数组函数对象SymbolSetWeakSetMapWeakMapProxyreflectProxy与Reflex结合实例classpromiseiteratorGerneratorDecorators模块学习资料 let /* let 声明变量 *//* es6相对于es5的全局和局部作用域&#xff0c;多了一个块作用域&#xff0c;块作用域里声明的…

jquery插件封装指南

入门 编写一个jQuery插件开始于给jQuery.fn加入​​新的功能属性&#xff0c;此处添加的对象属性的名称就是你插件的名称&#xff1a; jQuery.fn.myPlugin function(){//你自己的插件代码};用户非常喜欢的$符号哪里去了&#xff1f; 它仍然存在&#xff0c;但是&#xff0c;为…

synchronize原理

synchronized的三种应用方式 一. 修饰实例方法&#xff0c;作用于当前实例加锁&#xff0c;进入同步代码前要获得当前实例的锁。 二. 修饰静态方法&#xff0c;作用于当前类对象加锁&#xff0c;进入同步代码前要获得当前类对象的锁。 三. 修饰代码块&#xff0c;指定加锁对象&…

不能修改“System Roots”钥匙串

iOS mac添加证书 不能修改“System Roots”钥匙串错误 如图&#xff1a; 解决方式&#xff1a; 打开钥匙串---登录---&#xff0c;直接把证书拖过来 然后&#xff0c;查看--我的证书&#xff0c;里面&#xff0c;找到证书&#xff0c;即可

stylus在vue中的使用

stylus是一个css预处理器&#xff0c;比较流行的css预处理器有sass、less、stylus&#xff0c;它们都一样&#xff0c;都是css的语法糖&#xff0c;可以使用变量&#xff0c;可以有简单的逻辑&#xff0c;使css的开发效率更高&#xff0c;更易维护。stylus来自node社区&#xf…

未来产品的设计

Donald A. Norman继《情感化设计》之后&#xff0c;又一设计精品力作&#xff1a; 未来产品的设计样章试读及本书预定&#xff1a;http://www.china-pub.com/195642市场价 &#xff1a;&#xffe5;39.00 会员价 &#xff1a; &#xffe5;29.25(75折) 【作  者】(美)Donald…

vue-cli webpack 配置分析

目录结构 . ├── README.md ├── build │ ├── build.js │ ├── check-versions.js │ ├── dev-client.js │ ├── dev-server.js │ ├── utils.js │ ├── webpack.base.conf.js │ ├── webpack.dev.conf.js │ └── webpack.prod.c…

css之字体图标

SVG与字体图标 SVG图片是矢量图片&#xff0c;不会随着图片的伸缩而影响质量&#xff0c;通常把只有一种颜色的图标做成SVG&#xff0c;通过SVG生成字体图标&#xff0c;放到项目中使用。 https://icomoon.io/是一个比较快捷的生成字体图标的线上工具&#xff0c;进入主页后&…

history.back(-1)和history.go(-1)的区别

history.back(-1):直接返回当前页的上一页&#xff0c;数据全部消息&#xff0c;是个新页面 history.go(-1):也是返回当前页的上一页&#xff0c;不过表单里的数据全部还在 history.back(0) 刷新 history.back(1) 前进 history.back(-1) 后退

研发阶段模拟接口数据

因为在vue-cli工程中需要创建很多.vue文件&#xff0c;我们希望创建vue文件和创建html、css、js文件一样右键即可选择创建&#xff0c;并且创建的文件中可预先写好模板代码。 webstorm-Preferences打开选项界面 选择File and Code Templates&#xff0c;点击绿色加号 填…

真实项目中 ThreadLocal 的妙用

一、什么是 ThreadLocal ThreadLocal 提供了线程的局部变量&#xff0c;每个线程都可以通过 set() 和 get() 来对这个局部变量进行操作&#xff0c;但不会和其他线程的局部变量冲突&#xff0c;实现了线程间的据隔离。 简单讲&#xff1a;一个获取用户的请求线程 A&#xff0c;…