POST请求的消息主体放在entity body中,服务端根据请求头中的Content-Type字段来获取消息主体的编码方式,进而进行解析数据。

一、application/x-www-form-urlencoded

最常见的 POST 提交数据的方式,原生Form表单,如果不设置 enctype 属性,默认为application/x-www-form-urlencoded 方式提交数据。

首先,Content-Type被指定为 application/x-www-form-urlencoded;

其次,提交的表单数据会转换为键值对并按照 key1=val1&key2=val2 的方式进行编码,key 和 val 都进行了 URL 转码。大部分服务端语言都对这种方式有很好的支持。

另外,如利用AJAX 提交数据时,也可使用这种方式。例如 jQuery,Content-Type 默认值都是”application/x-www-form-urlencoded;charset=utf-8”。

1
2
3
4
5
curl --location --request POST 'http://127.0.0.1:5000/test/getData?p1=1&p2=2' \
--header 'Content-Type: application/x-www-form-urlencoded' \
--data-urlencode 'x1=xv1' \
--data-urlencode 'x2=xv2' \
--data-urlencode 'x3=xv3'
1
2
3
4
5
6
POST /test/getData?p1=1&p2=2 HTTP/1.1
Host: 127.0.0.1:5000
Content-Type: application/x-www-form-urlencoded
Content-Length: 20

x1=xv1&x2=xv2&x3=xv3
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
import requests

url = "http://127.0.0.1:5000/test/getData?p1=1&p2=2"

payload='x1=xv1&x2=xv2&x3=xv3'
headers = {
  'Content-Type': 'application/x-www-form-urlencoded'
}

response = requests.request("POST", url, headers=headers, data=payload)

print(response.text)

二、multipart/form-data

另一个常见的 POST 数据提交的方式, Form 表单的 enctype 设置为multipart/form-data,它会将表单的数据处理为一条消息,以标签为单元,用分隔符(这就是boundary的作用)分开,类似我们上面Content-Type中的例子。   由于这种方式将数据有很多部分,它既可以上传键值对,也可以上传文件,甚至多个文件。

当上传的字段是文件时,会有Content-Type来说明文件类型;Content-disposition,用来说明字段的一些信息。 每部分都是以 –boundary 开始,紧接着是内容描述信息,然后是回车,最后是字段具体内容(字段、文本或二进制等)。如果传输的是文件,还要包含文件名和文件类型信息。消息主体最后以 –boundary– 标示结束。

1
2
3
4
5
curl --location --request POST 'http://127.0.0.1:5000/test/getData?p1=1&p2=2' \
--header 'Content-Type: multipart/form-data' \
--form 'f1="formA"' \
--form 'f2="formB"' \
--form 'f3="formC"'
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
POST /test/getData?p1=1&p2=2 HTTP/1.1
Host: 127.0.0.1:5000
Content-Type: multipart/form-data; boundary=----WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Length: 299

----WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Disposition: form-data; name="f1"

formA
----WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Disposition: form-data; name="f2"

formB
----WebKitFormBoundary7MA4YWxkTrZu0gW
Content-Disposition: form-data; name="f3"

formC
----WebKitFormBoundary7MA4YWxkTrZu0gW
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
import requests

url = "http://127.0.0.1:5000/test/getData?p1=1&p2=2"

payload={
    'f1': 'formA',
    'f2': 'formB',
    'f3': 'formC'
}
files=[

]
headers = {
  'Content-Type': 'multipart/form-data'
}

response = requests.request("POST", url, headers=headers, data=payload, files=files)

print(response.text)

三、application/json

Content-Type: application/json 作为响应头比较常见。实际上,现在越来越多的人把它作为请求头,用来告诉服务端消息主体是序列化后的 JSON 字符串,其中一个好处就是JSON 格式支持比键值对复杂得多的结构化数据。由于 JSON 规范的流行,除了低版本 IE 之外的各大浏览器都原生支持JSON.stringify,服务端语言也都有处理 JSON 的函数,使用起来没有困难。 Google 的 AngularJS 中的 Ajax 功能,默认就是提交 JSON 字符串。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
curl --location --request POST 'http://127.0.0.1:5000/test/getData' \
--header 'Content-Type: application/json' \
--data-raw '{
    "article_title":"斗罗大陆",
    "f_type":2,
    "article_content":"海神岛",
    "article_url":"http://www.baidu.com/3.jpg",
    "filelist":[
        "one.txt","two.txt","three.txt"
    ],
    "source":{
        "a":1,
        "b":2
    }
}'
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
POST /test/getData?p1=1&p2=2 HTTP/1.1
Host: 127.0.0.1:5000
Content-Type: application/json
Content-Length: 238

{
    "article_title":"斗罗大陆",
    "f_type":2,
    "article_content":"海神岛",
    "article_url":"http://www.baidu.com/3.jpg",
    "filelist":[
        "one.txt","two.txt","three.txt"
    ],
    "source":{
        "a":1,
        "b":2
    }
}
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
import requests

url = "http://127.0.0.1:5000/test/getData?p1=1&p2=2"

#字符串类型
payload="{\n    \"article_title\": \"斗罗大陆\",\n    \"f_type\": 2,\n    \"article_content\": \"海神岛\",\n    \"article_url\": \"http://www.baidu.com/3.jpg\",\n    \"filelist\": [\n        \"one.txt\",\n        \"two.txt\",\n        \"three.txt\"\n    ],\n    \"source\": {\n        \"a\": 1,\n        \"b\": 2\n    }\n}"

headers = {
  'Content-Type': 'application/json'
}

response = requests.request("POST", url, headers=headers, data=payload)

print(response.text)

指定其他类型:

  • Content-Type: text/plain
  • Content-Type: text/html
  • Content-Type: application/xml
  • Content-Type: application/javascript

四、text/xml

XML的作用不言而喻,用于传输和存储数据,它非常适合万维网传输,提供统一的方法来描述和交换独立于应用程序或供应商的结构化数据,在JSON出现之前是业界一大标准(当然现在也是),相比JSON的优缺点大家有兴趣可以上网search。因此,在POST提交数据时,xml类型也是不可缺少的一种,虽然一般场景上使用JSON可能更轻巧、灵活。

五、binary (application/octet-stream)

在Chrome浏览器的Postman工具中,还可以看到”binary“这一类型,指的就是一些二进制文件类型。如application/pdf,指定了特定二进制文件的MIME类型。就像对于text文件类型若没有特定的子类型(subtype),就使用 text/plain。类似的,二进制文件没有特定或已知的 subtype,即使用 application/octet-stream,这是应用程序文件的默认值,一般很少直接使用 。

对于application/octet-stream,只能提交二进制,而且只能提交一个二进制,如果提交文件的话,只能提交一个文件,后台接收参数只能有一个,而且只能是流(或者字节数组)。

很多web服务器使用默认的 application/octet-stream 来发送未知类型。出于一些安全原因,对于这些资源浏览器不允许设置一些自定义默认操作,导致用户必须存储到本地以使用。一般来说,设置正确的MIME类型很重要。

其他二进制类型(图片,pdf,word,ppt,视频,音频):

  • image/jpeg
  • image/x-icon
  • image/gif
  • application/pdf
  • application/msword
  • application/x-ppt
  • video/mpg
  • video/mpeg
  • video/mpeg4
  • video/avi
  • audio/basic
  • audio/mpegurl
  • audio/mp3
  • audio/mid
  • audio/wav