背景:
Http请求时,如果是get请求,那么表单参数以name=value&name1=value1的形式附到url的后面,如果是post请求,那么表单参数是在请求体中,也是以name=value&name1=value1的形式在请求体中(可以通过chrome的开发者工具看出)。
Content-Type
1、application/x-www-form-urlencoded 在发送前编码所有字符(默认)
2、multipart/form-data 不对字符编码。在使用包含文件上传控件的表单时,必须使用该值。
3、text/plain 空格转换为 “+” 加号,但不对特殊字符编码。
POST原生ajax请求
不设置content-type时,默认为text/plain;charset=UTF-8,请求参数在request payload中,但通过nodejs获取到的req.body却是空的。
查看资料后才发现,content-type不是application/x-www-form-urlencoded的post请求是不会读取请求体数据并进行相应处理的,故body-parser模块不会解析。所以在利用原生ajax的时候要设置content-type来告知服务器如何解析参数。
思考
form表单之所以不会出现这个问题是因为,表单数据默认使用”application/x-www-form-urlencoded”。就是说,在发送到服务器之前,所有字符都会进行编码(空格转换为 “+” 加号,特殊符号转换为 ASCII HEX 值)。
服务器为什么会对表单提交和文件上传做特殊处理,因为表单提交数据是名值对的方式,且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,这时服务器就不知道怎么解析数据了,所以才只能通过获取原始数据流的方式来进行解析请求数据。