You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

193 lines
13 KiB
Markdown

2 years ago
# 11 | 你能写出正确的网址吗?
上一讲里我们一起学习了HTTP协议里的请求方法其中最常用的一个是GET它用来从服务器上某个资源获取数据另一个是POST向某个资源提交数据。
那么,应该用什么来标记服务器上的资源呢?怎么区分“这个”资源和“那个”资源呢?
经过前几讲的学习你一定已经知道了用的是URI也就是**统一资源标识符****U**niform **R**esource **I**dentifier。因为它经常出现在浏览器的地址栏里所以俗称为“网络地址”简称“网址”。
严格地说URI不完全等同于网址它包含有URL和URN两个部分在HTTP世界里用的网址实际上是URL——**统一资源定位符****U**niform **R**esource **L**ocator。但因为URL实在是太普及了所以常常把这两者简单地视为相等。
不仅我们生活中的上网要用到URI平常的开发、测试、运维的工作中也少不了它。
如果你在客户端做iOS、 Android或者某某小程序开发免不了要连接远程服务就会调用底层API用URI访问服务。
如果你使用Java、PHP做后台Web开发也会调用getPath()、parse\_url() 等函数来处理URI解析里面的各个要素。
在测试、运维配置Apache、Nginx等Web服务器的时候也必须正确理解URI分离静态资源与动态资源或者设置规则实现网页的重定向跳转。
总之一句话URI非常重要要搞懂HTTP甚至网络应用就必须搞懂URI。
## URI的格式
不知道你平常上网的时候有没有关注过地址栏里的那一长串字符,有的比较简短,有的则一行都显示不下,有的意思大概能看明白,而有的则带着各种怪字符,有如“天书”。
其实只要你弄清楚了URI的格式就能够轻易地“破解”这些难懂的“天书”了。
URI本质上是一个字符串这个字符串的作用是**唯一地标记资源的位置或者名字**。
这里我要提醒你注意它不仅能够标记万维网的资源也可以标记其他的如邮件系统、本地文件系统等任意资源。而“资源”既可以是存在磁盘上的静态文本、页面数据也可以是由Java、PHP提供的动态服务。
下面的这张图显示了URI最常用的形式由scheme、host:port、path和query四个部分组成但有的部分可以视情况省略。
![](https://static001.geekbang.org/resource/image/46/2a/46581d7e1058558d8e12c1bf37d30d2a.png)
## URI的基本组成
URI第一个组成部分叫**scheme**,翻译成中文叫“**方案名**”或者“**协议名**”,表示**资源应该使用哪种协议**来访问。
最常见的当然就是“http”了表示使用HTTP协议。另外还有“https”表示使用经过加密、安全的HTTPS协议。此外还有其他不是很常见的scheme例如ftp、ldap、file、news等。
浏览器或者你的应用程序看到URI里的scheme就知道下一步该怎么走了会调用相应的HTTP或者HTTPS下层API。显然如果一个URI没有提供scheme即使后面的地址再完善也是无法处理的。
在scheme之后必须是**三个特定的字符**“**://**”它把scheme和后面的部分分离开。
实话实说,这个设计非常的怪异,我最早上网的时候看见地址栏里的“://”就觉得很别扭直到现在也还是没有太适应。URI的创造者蒂姆·伯纳斯-李也曾经私下承认“://”并非必要,当初有些“过于草率”了。
不过这个设计已经有了三十年的历史,不管我们愿意不愿意,只能接受。
在“://”之后,是被称为“**authority**”的部分,表示**资源所在的主机名**通常的形式是“host:port”即主机名加端口号。
主机名可以是IP地址或者域名的形式必须要有否则浏览器就会找不到服务器。但端口号有时可以省略浏览器等客户端会依据scheme使用默认的端口号例如HTTP的默认端口号是80HTTPS的默认端口号是443。
有了协议名和主机地址、端口号,再加上后面**标记资源所在位置**的**path**,浏览器就可以连接服务器访问资源了。
URI里path采用了类似文件系统“目录”“路径”的表示方式因为早期互联网上的计算机多是UNIX系统所以采用了UNIX的“/”风格。其实也比较好理解它与scheme后面的“://”是一致的。
这里我也要再次提醒你注意URI的path部分必须以“/”开始,也就是必须包含“/”,不要把“/”误认为属于前面authority。
说了这么多“理论”,来看几个实例。
```
http://nginx.org
http://www.chrono.com:8080/11-1
https://tools.ietf.org/html/rfc7230
file:///D:/http_study/www/
```
第一个URI算是最简单的了协议名是“http”主机名是“nginx.org”端口号省略所以是默认的80而路径部分也被省略了默认就是一个“/”,表示根目录。
第二个URI是在实验环境里这次课程的专用URI主机名是“www.chrono.com”端口号是8080后面的路径是“/11-1”。
第三个是HTTP协议标准文档RFC7230的URI主机名是“tools.ietf.org”路径是“/html/rfc7230”。
最后一个URI要注意了它的协议名不是“http”而是“file”表示这是本地文件而后面居然有三个斜杠这是怎么回事
如果你刚才仔细听了scheme的介绍就能明白这三个斜杠里的前两个属于URI特殊分隔符“://”,然后后面的“/D:/http\_study/www/”是路径而中间的主机名被“省略”了。这实际上是file类型URI的“特例”它允许省略主机名默认是本机localhost。
但对于HTTP或HTTPS这样的网络通信协议主机名是绝对不能省略的。原因之前也说了会导致浏览器无法找到服务器。
我们可以在实验环境里用Chrome浏览器再仔细观察一下HTTP报文里的URI。
运行Chrome用F12打开开发者工具然后在地址栏里输入“[http://www.chrono.com/11-1](http://www.chrono.com/11-1)”,得到的结果如下图。
![](https://static001.geekbang.org/resource/image/20/9f/20ac5ee55b8ee30527492c8abb60ff9f.png)
在开发者工具里依次选“Network”“Doc”就可以找到请求的URI。然后在Headers页里看Request Headers用“view source”就可以看到浏览器发的原始请求头了。
发现了什么特别的没有?
在HTTP报文里的URI“/11-1”与浏览器里输入的“[http://www.chrono.com/11-1](http://www.chrono.com/11-1)”有很大的不同,协议名和主机名都不见了,只剩下了后面的部分。
这是因为协议名和主机名已经分别出现在了请求行的版本号和请求头的Host字段里没有必要再重复。当然在请求行里使用完整的URI也是可以的你可以在课后自己试一下。
通过这个小实验我们还得到了一个结论客户端和服务器看到的URI是不一样的。客户端看到的必须是完整的URI使用特定的协议去连接特定的主机而服务器看到的只是报文请求行里被删除了协议名和主机名的URI。
如果你配置过Nginx你就应该明白了Nginx作为一个Web服务器它的location、rewrite等指令操作的URI其实指的是真正URI里的path和后续的部分。
## URI的查询参数
使用“协议名+主机名+路径”的方式,已经可以精确定位网络上的任何资源了。但这还不够,很多时候我们还想在操作资源的时候附加一些额外的修饰参数。
举几个例子获取商品图片但想要一个32×32的缩略图版本获取商品列表但要按某种规则做分页和排序跳转页面但想要标记跳转前的原始页面。
仅用“协议名+主机名+路径”的方式是无法适应这些场景的所以URI后面还有一个“**query**”部分它在path之后用一个“?”开始,但不包含“?”,表示对资源附加的额外要求。这是个很形象的符号,比“://”要好的多,很明显地表示了“查询”的含义。
查询参数query有一套自己的格式是多个“**key=value**”的字符串这些KV值用字符“**&**”连接,浏览器和服务器都可以按照这个格式把长串的查询参数解析成可理解的字典或关联数组形式。
你可以在实验环境里用Chrome试试下面这个加了query参数的URI
```
http://www.chrono.com:8080/11-1?uid=1234&name=mario&referer=xxx
```
Chrome的开发者工具也能解码出query里的KV对省得我们“人肉”分解。
![](https://static001.geekbang.org/resource/image/e4/f3/e42073080968e8e0c58d9a9126ab82f3.png)
还可以再拿一个实际的URI来看一下这个URI是某电商网站的一个商品查询URI比较复杂但相信现在的你能够毫不费力地区分出里面的协议名、主机名、路径和查询参数。
```
https://search.jd.com/Search?keyword=openresty&enc=utf-8&qrst=1&rt=1&stop=1&vt=2&wq=openresty&psort=3&click=0
```
你也可以把这个URI输入到Chrome的地址栏里再用开发者工具仔细检查它的组成部分。
## URI的完整格式
讲完了query参数URI就算完整了HTTP协议里用到的URI绝大多数都是这种形式。
不过必须要说的是URI还有一个“真正”的完整形态如下图所示。
![](https://static001.geekbang.org/resource/image/ff/38/ff41d020c7a27d1e8191057f0e658b38.png)
这个“真正”形态比基本形态多了两部分。
第一个多出的部分是协议名之后、主机名之前的**身份信息**“user:passwd@”表示登录主机时的用户名和密码但现在已经不推荐使用这种形式了RFC7230因为它把敏感信息以明文形式暴露出来存在严重的安全隐患。
第二个多出的部分是查询参数后的**片段标识符**“#fragment”它是URI所定位的资源内部的一个“锚点”或者说是“标签”浏览器可以在获取资源后直接跳转到它指示的位置。
但片段标识符仅能由浏览器这样的客户端使用,服务器是看不到的。也就是说,浏览器永远不会把带“#fragment”的URI发送给服务器服务器也永远不会用这种方式去处理资源的片段。
## URI的编码
刚才我们看到了在URI里只能使用ASCII码但如果要在URI里使用英语以外的汉语、日语等其他语言该怎么办呢
还有某些特殊的URI会在path、query里出现“@&?"等起界定符作用的字符会导致URI解析错误这时又该怎么办呢
所以URI引入了编码机制对于ASCII码以外的字符集和特殊字符做一个特殊的操作把它们转换成与URI语义不冲突的形式。这在RFC规范里称为“escape”和“unescape”俗称“转义”。
URI转义的规则有点“简单粗暴”直接把非ASCII码或特殊字符转换成十六进制字节值然后前面再加上一个“%”。
例如,空格被转义成“%20”“?”被转义成“%3F”。而中文、日文等则通常使用UTF-8编码后再转义例如“银河”会被转义成“%E9%93%B6%E6%B2%B3”。
有了这个编码规则后URI就更加完美了可以支持任意的字符集用任何语言来标记资源。
不过我们在浏览器的地址栏里通常是不会看到这些转义后的“乱码”的这实际上是浏览器一种“友好”表现隐藏了URI编码后的“丑陋一面”不信你可以试试下面的这个URI。
```
http://www.chrono.com:8080/11-1?夸父逐日
```
先在Chrome的地址栏里输入这个query里含有中文的URI然后点击地址栏把它再拷贝到其他的编辑器里它就会“现出原形”
```
http://www.chrono.com:8080/11-1?%E5%A4%B8%E7%88%B6%E9%80%90%E6%97%A5
```
## 小结
今天我们学习了网址也就是URI的知识在这里小结一下今天的内容。
1. URI是用来唯一标记服务器上资源的一个字符串通常也称为URL
2. URI通常由scheme、host:port、path和query四个部分组成有的可以省略
3. scheme叫“方案名”或者“协议名”表示资源应该使用哪种协议来访问
4. “host:port”表示资源所在的主机名和端口号
5. path标记资源所在的位置
6. query表示对资源附加的额外要求
7. 在URI里对“@&/”等特殊字符和汉字必须要做编码否则服务器收到HTTP报文后会无法正确处理。
## 课下作业
1. HTTP协议允许在在请求行里使用完整的URI但为什么浏览器没有这么做呢
2. URI的查询参数和头字段很相似都是key-value形式都可以任意自定义那么它们在使用时该如何区别呢具体分析可以在“答疑篇”[第41讲](https://time.geekbang.org/column/article/146833)中的URI查询参数和头字段部分查看
欢迎你把自己的答案写在留言区,与我和其他同学一起讨论。如果你觉得有所收获,欢迎你把文章分享给你的朋友。
![](https://static001.geekbang.org/resource/image/26/cb/26e06ff1fee9a7cd0b9c1a99bf9d32cb.png)