本节来介绍一下 http 请求,请求是由客户端向服务器发送的,一般可以分为请求行、请求头、空行和请求体四个部分,如下图所示:
图:http请求结构
1. 请求行
请求行是由请求方法、请求 uri(uri 全称为 universal resource identifier,中文含义为“统一资源标志符”,是用来标识抽象或物理资源的字符串)和 http 协议版本三个部分组成,每个部分使用空格分隔,在请求行的最后以回车符与换行符结尾。如下所示:
get /script/common.js http/1.1
其中,get 为请求方法,/script/common.js 为请求 uri,http/1.1 为 http 协议的版本。
请求方法用来表示使用哪种方式来请求由请求 uri 所指向的资源。注意,请求方法始终是以大写形式存在的。
http 协议中常用的请求方法如下表所示:
方法 | 描述 |
---|---|
get | 请求指定 uri 所指向的资源,并返回 |
head | 与 get 请求类似,但只获取由 uri 所指向资源的响应消息报头 |
post | 将数据提交到服务器(例如提交表单或者上传文件),数据被包含在请求体中 |
put | 使用从客户端向服务器传送的数据替换指定文档的内容 |
delete | 请求服务器删除 uri 所指向的目标资源 |
connect | http/1.1 协议中预留的能够将连接改为管道方式的代理服务器 |
options | 允许客户端查看服务器的性能 |
trace | 回显服务器收到的请求,主要用于测试或诊断 |
patch | 是对 put 方法的补充,用来对已知资源进行局部更新 |
2. 请求头
请求头是客户端传递给服务器的一系列有关本次请求和客户端本身的相关信息。请求头一般由头部字段名、冒号(:)、空格、值组成,例如host: c.biancheng.net
。
http 协议中常用的头部字段名以及含义如下表所示:
头部字段名 | 说明 | 示例 |
---|---|---|
accept | 指定客户端能够接收的内容类型 | accept: text/plain, text/html |
accept-charset | 浏览器可以接受的字符编码集 | accept-charset: iso-8859-5 |
accept-encoding | 指定浏览器支持的内容压缩类型 | accept-encoding: compress, gzip |
accept-language | 浏览器可以接受的语言 | accept-language: en,zh |
authorization | http 授权的授权证书 | authorization: basic qwxhzgrpbjpvcgvuihnlc2ftzq== |
cache-control | 指定请求和响应遵循的缓存机制 | cache-control: no-cache |
connection | 表示是否需要长连接。(http 1.1 默认进行长连接) | connection: close |
cookie | http 请求发送时,会把保存在该请求域名下的所有 cookie 值一起发送给 web 服务器 | cookie: $version=1; skin=new; |
content-length | 请求内容的长度 | content-length: 348 |
content-type | 请求的 mime 类型 | content-type: application/x-www-form-urlencoded |
date | 请求发送的日期和时间 | date: tue, 15 nov 2010 08:12:31 gmt |
expect | 请求特定的服务器行为 | expect: 100-continue |
from | 发起请求的用户 email | from: user@email.com |
host | 指定请求服务器的域名和端口号 | host: c.biancheng.net |
if-match | 只有请求内容与实体相匹配才有效 | if-match: “737060cd8c284d8af7ad3082f209582d” |
if-modified-since | 如果请求的部分在指定时间之后被修改则请求成功,未被修改则返回 304 代码 | if-modified-since: sat, 29 oct 2010 19:43:31 gmt |
if-none-match | 如果内容未改变返回 304 代码,参数为服务器先前发送的 etag(被请求变量的实体值),与服务器回应的 etag 比较,判断是否改变 | if-none-match: “737060cd8c284d8af7ad3082f209582d” |
if-range | 如果实体未改变,服务器发送客户端丢失的部分,否则发送整个实体。参数也为 etag | if-range: “737060cd8c284d8af7ad3082f209582d” |
if-unmodified-since | 如果实体在指定时间之后未被修改则请求成功 | if-unmodified-since: sat, 29 oct 2010 19:43:31 gmt |
max-forwards | 限制信息通过代理和网关传送的时间 | max-forwards: 10 |
pragma | 用来包含实现特定的指令 | pragma: no-cache |
proxy-authorization | 连接到代理的授权证书 | proxy-authorization: basic qwxhzgrpbjpvcgvuihnlc2ftzq== |
range | 只请求实体的一部分,用来指定范围 | range: bytes=500-999 |
referer | 先前网页的地址,即来路 | referer: http://c.biancheng.net/php/ |
te | 客户端愿意接受的传输编码 | te: trailers,deflate;q=0.5 |
upgrade | 向服务器指定某种传输协议以便服务器进行转换(如果支持) | upgrade: http/2.0, shttp/1.3, irc/6.9, rta/x11 |
user-agent | 发出请求的客户端信息 | user-agent: mozilla/5.0 (linux; x11) |
via | 通知中间网关或代理服务器地址,通信协议 | via: 1.0 fred, 1.1 nowhere.com (apache/1.1) |
warning | 关于消息实体的警告信息 | warn: 199 miscellaneous warning |
3. 空行
请求头结束后再往下是一个由回车符和换行符构成的空行,它没有什么特殊的含义,主要是用来做请求头的结束标识符。
4. 请求体
请求体一般出现在使用 post 方法提交数据的时候,比如提交表单数据或者上传文件时。客户端会把数据以“key=value”的形式发送给服务器,多条数据之间使用“&”符号分隔,如下所示:
username=c.biancheng.net&password=123456
5. http请求示例
将上面的几个部分组合到一起就构成了一个 http 请求,下面以访问 c.biancheng.net 上的 hello.htm 页面为例,http 请求如下所示:
get /hello.htm http/1.1
user-agent: mozilla/4.0 (compatible; msie5.01; windows nt)
host: c.biancheng.net
accept-language: en-us
accept-encoding: gzip, deflate
connection: keep-alive
由于对服务器的请求中没有提交任何请求数据,所以请求中没有请求体。下面的示例演示了客户端在提交表单数据时向服务器发送的 http 请求,如下所示:
post /login.php http/1.1
user-agent: mozilla/4.0 (compatible; msie5.01; windows nt)
host: c.biancheng.net
content-type: application/x-www-form-urlencoded
content-length: length
accept-language: en-us
accept-encoding: gzip, deflate
connection: keep-alive
username=c.biancheng.net&password=123456
此处,给定的 url /login.php 将用于处理提交的数据,并做出响应。这里 content-type 告诉服务器所传递的数据是简单的 web 表单数据,而 length 则是数据的实际长度。