成为一个合格的前端工程师 - HTTP - 2

2017-07-30  本文已影响9人  ahole

前言

上一篇 HTTP 讲到了第3章。这次我们讲第4章。有太多要学,不是科班出生的表示压力有点大,需要赶紧学,幸运的是我现在这家公司每个人我都能在他们身上学到东西。

TCP 是啥

HTTP的上一层,传输层用的是TCP ,书中简单的说了下TCP,建议大家还是去看看 《TCP/IP 详解一卷》。那个说的详细。

HTTP 事务的时延

我们发送一个请求,其实在服务器处理的时间一般都不长,时间上的花费是在传输数据的路上。见下图:


一条请求的时间

TCP连接的握手时延

握手需要发送三个TCP数据,这些是发送HTTP数据的准备工作花费了很多时间,原本在发送3个TCP包前是不会发送HTTP请求的,但是后来发现在TCP握手的最后一个数据包可以利用发送数据,这种技术叫做捎带。请求就变成这个样子:

注意ACK已经开始发送HTTP请求了

延迟确认

书中表面:TCP传输的时候都要接收方回发一个已收到的确认信息(确认报文),来告诉发送方,信息已经收到,不需要再发送。
一个确认报文是很小的,如果单独作为一个TCP包发送显得太浪费,所以操作系统会使用延时确认技术,简单的说就是在确认收到信息的时候,不立马发送确认报文,而是等待一小段时间,等待有同地址的报文,并将其合并一起发送。类比现实生活很是形象。

TCP慢启动

简单的说就是: 开始发送TCP请求时不会立马发送多个TCP报文,而是慢慢的并行的发送数量。比如成功发送一个之后,一次发送两个,两个成功之后。。。这样是有好处的,比如可以避免多个TCP报文因为同一个原因发送失败的而引发的性能浪费。

Connection

Connection 是HTTP的一个首部字段。
特点一:在Connection中包含的字段,不能被转出到下一个地址。比如 Connection: name .那么name 这个首部就只能到下一个地址,活不到被下一个转出。

请求方式

下面我们来聊聊web怎么玩HTTP

串行

这是最原始的时代,估计比我还老。这种方式就是发送一个HTTP请求等其全部完成之后再发送下一个请求。(stupid)
直接上图吧,不哔哔:

连接-* 这里是建立TCP连接

并行

谁动能行到能并行,计算机本来就用的特性。
上例🌰的情况,我们分析,可以发现,1. 一次只发一个请求,浪费带宽。2. 事务之间的联系不一定要有先后关系,可以同时请求相互不影响。

模型图:( 书中给的例子是一个HTML 和 3个图片)

典型的页面加载过程

并行的问题:

  1. 带宽可能有限,带宽让我们不可能高效的并行多个请求。
  2. 性能,切换进程是需要花费性能的。
  3. 服务器拒绝,你的机子牛逼了,服务器不干了。大家都一次发这么多请求,服务器撑不住的。你慢点。😈

持久连接

并行连接其实并没有本质的提高连接的速度。它只是利用了计算机的进程功能,并没有利用网络知识。
非持续连接的问题:每次完成HTTP请求之后都要关闭TCP连接,但是我们的网站可能对同一个服务器发送多个请求,所以花费了很多时间在开启关闭TCP连接的过程中。
HTTP1.1 就通过网络知识本质的提升了性能。
持续连接:在一个HTTP请求完成之后一段时间内不关闭TCP的连接。
HTTP持久连接解决的问题:

  1. 减少建立TCP连接的时延
  2. 减少慢启动的时延

上图:

书中图画的不好,没有把第一次建立连接,和慢启动的优化显示出来

http1.0 持久连接 - Keep-alive

http1.0 如果想使用持久连接,会在请求报文中加入 Connection:keep-Alive ,如果服务器同意就会回发Connection:keep-alive .
一般 Connection:keep-alive 会和 keep-alive 首部一起使用 。如

Connection: Keep-Alive
Keep-Alive: max=5, timeout=120 ### 服务器还能支持保留多少个连接, 希望保持120秒的连接
keep-alive 的规则与限制
  1. 持久连接建立起来之后,如果在这条连接上没有在继续发 Connection:Keep-Alive 那么持久连接就会断开。
  2. 实体的主题部分必须代用正确的Content-Length,这样就能让另一端知道是报文的结束还是新报文的开始。占坑不是太懂
  3. 哑代理问题
哑代理

哑代理的问题会出现在我们的网络路线中出现一个不懂Keep-Alive机制又盲目转发的代理的情况。
首先我们主机发送一份Keep-Alive的报文,代理接受到这样一份报文,它不懂客户端要保持连接,然后又转发请求给服务器,服务器懂啊,就回了一份Connection:Keep-Alive的报文表示同意保持连接,中间代理又接受到了这样的报文,还是不懂啊,再转发给客户端。这是客户端和服务端都懂对方😝。那么客户端下一次就不会再发TCP握手数据啦,直接发请求。但是这时代理就不干了,你不发建立连接的请求,不符合规矩,直接无视。尴尬的哑代理情况就出现了。
这种尴尬的情况书中没有给出完备的解决方案。

HTTP1.1 默认持续连接

HTTP1.1 持久连接的规章制度
  1. HTTP1.1 不用发送Connection:keep-Alive 也是持久连接的
  2. 只有在发送Connection: close 时才能断开持久连接
  3. 实体主体部分的长度与相应的Content-Length一致,或者使用分块出函数编码方式。连接才能持久保持。占坑
  4. HTTP1.1 代理必须能够分别管理客户端和服务器的持久连接
  5. 一个客户端最多只能与服务器或代理保持最多两个持久连接

管道化连接

管道化在我的理解就是并行版的持久化连接请求。
就是在建立持久化连接后,在不等待上一条请求成功返回前就发送下一条请求。就是这么简单。
上图:

(b)为持久化串行请求

书中说了4点管道化的注意事项,但我感觉太过啰嗦,都是些开发者自己应该注意的问题,不在我们的知识学习范围内不说了。

关于关闭连接

书中建议我们不适用管道化来发一些非幂等的请求,因为如果中间断开了很难判断是否服务端处理了请求,还是还没收到就已经断开了连接。

记录生活

我又恋爱了,又是那个女孩。异地恋。反而让我觉得我应该更快的成熟。原本想着毕业第一年好好的善待自己,看来还是要规划下那少的可怜的薪水。

上一篇 下一篇

猜你喜欢

热点阅读