一、短连接
这里的连接指的是 TCP 连接。一个 TCP 连接从创建到结束一共有 3 个阶段,分别为“三次握手”建立连接、客户端与服务端进行数据包传输、“四次挥手”断开连接。
客户端与服务端的每一次完整的消息交互(发请求——响应)都建立一次 TCP 连接,当这次交互完毕后就释放该 TCP 连接。这个过程就是短连接。
早期的 http 1.0 用的就是短连接。
- 优点:简单。因为存在的连接都是正在通信的有用连接,不需要过多的管理。
- 缺点:浪费资源,网络延迟较大。
为什么有这样的缺点呢?
通常来说,一个网页的加载是需要向服务端发送大量的资源请求的,包括 js 文件请求、css 样式文件请求、各种图片请求、加载页面数据的异步请求、还可能存在音频视频等媒体资源请求等等。这么多次的同一个客户端到同一个服务端的请求每次都要建立 TCP 就显得有点多余。
以通话为例,一次通话就是一次 TCP 连接,一个对话就是一次信息交互。我们可能每次通话只进行一个对话就挂线、下个对话再再拨一次号吗?显然不可能,因为无意义;拨号需要时间和成本,拨号次数越多,花费时间和成本就越高,会造成无意义的浪费。TCP 连接的“三次握手”和“四次挥手”即消耗时间,也浪费资源。
二、长连接
长连接是在 http 1.1 中提出并默认使用的。它最大的特点的就是 TCP 连接能够保持一段时间(超过这个时间会自动断开),不会再一次信息交互后马上断开,下一个请求会继续使用该 TCP 连接,达到 TCP 连接复用的效果。在 http 1.1 协议中,在响应头里用 “Connect : keep-alive” 来表示长连接。
- 优点:有效复用 TCP 连接,减少网络延迟。
- 缺点:需要对每个 TCP 连接增加管理,占用服务器的更多的内存。因为 TCP 连接能够保持一段时间,所以需要判断该 TCP 连接是否失效、是否应该释放连接;无论 TCP 连接是否正处于通信状态,只要是在有效期内的都要存储。
三、短轮询
轮询指的是客户端每隔一段时间就向服务端发送一次请求,最典型的例子是获取最新聊天消息。
短轮询指的是,在轮询的过程中客户端每发一个请求,服务端都会返回结果。例如:
|
其实,在这种情景下,没消息也回复就显得有点冗余。假设,现在有很 10000 用户都在同一时刻进行短轮询,有新消息,服务端要进行 10000 个响应;由于并不是每一秒都会有新消息的,不如说没新消息的时间占比例更大,这时服务端也要进行 10000 个无新消息响应。这样会造成无意义的资源浪费。
四、长轮询
长轮询与短轮询的不同之处就在于,服务端在没新消息是是不会马上进行响应,而是将该请求挂起,直到有新消息时再响应,或者等到请求超时直接删除。客户端等到请求有结果(成功或失败),才发下一个请求。这样就有效解决了短轮询的问题。例如:
|
- 缺点:服务器将请求挂起也是要消耗资源的,而且返回数据顺序无保证。
五、流
实现服务器推送的方法除了轮询外,还可以用 http 流。不同于轮询,流 的生命周期内只使用一个 http 连接。
客户端发一个请求,服务器保持 http 连接打开,然后周期性向浏览器发送数据。
客户端的实现代码:
|
要说轮询是客户端向服务器催债的话,那么 http 流就是服务器向客户端分期付款。
六、SSE
SSE(Server-Sent Event,服务器发送事件)是服务器到客户端的单向连接模式,只允许服务器向客户端发送消息。
SSE 的 API 为 EventSource 对象。使用方法:
|
响应的 MIME 类型为 text/event-stream,响应格式为纯文本。每个数据项都有 data:前缀,如:
|
默认情况下,EventSource 对象会保持与服务器的连接。如果断开会自动重连;如果想永久断开,调用 sse.close() 方法。
七、Web Socket
Web Socket 是一种全双通、双向通信的持久连接。在 js 中创建 Web Socket 后,客户端会发一个 http 请求,取得服务器响应后,建立的连接会从 http 协议变为 Web Socket 协议。未加密连接从 http:// 变为 ws://,加密连接从 https:// 变为 wss://。
Web Socket 使用自定义协议的好处是,能够在客户端和服务器之间发送更少量的数据,而不必担心 http 的字节级开销。但是,由于自定义协议的缺陷,不断有人发现这个协议存在一致性和安全性问题。
Web Socket 使用方法:
|