上次更新于 2012 年 11 月 5 日(Spring MVC 3.2 RC1)
在我的上一篇文章中,我介绍了Spring MVC 3.2 中新的基于 Servlet 3 的异步支持,并讨论了长时间运行的请求。异步处理的第二个非常重要的动机是浏览器需要接收实时更新。示例包括在浏览器中聊天、股票行情、状态更新、实时体育赛事结果等。当然,并非所有示例都同样延迟敏感,但它们都具有类似的需求。
在标准的 HTTP 请求-响应语义中,浏览器发起请求,服务器发送响应,这意味着服务器只有在收到浏览器的请求后才能发送新信息。已经发展出几种方法,包括传统轮询、长轮询和HTTP 流,最近我们还有WebSocket协议。
传统轮询
浏览器不断发送请求以检查是否有新信息,服务器每次都会立即响应。这适合可以以合理稀疏的间隔进行轮询的场景。例如,邮件客户端可以每 10 分钟检查一次新邮件。它很简单,而且有效。但是,当必须尽快显示新信息时,这种方法效率低下,在这种情况下,轮询必须非常频繁。
长轮询
浏览器不断发送请求,但服务器只有在有新信息要发送时才会响应。从客户端的角度来看,这与传统轮询相同。从服务器的角度来看,这与长时间运行的请求非常相似,并且可以使用第 1 部分中讨论的技术进行扩展。
响应可以保持打开多长时间?浏览器设置为在 5 分钟后超时,并且网络中介(如代理)可以更早超时。因此,即使没有新的信息到达,长轮询请求也应该定期完成,以允许浏览器发送新的请求。这IETF 文档建议使用 30 到 120 秒之间的超时值,但实际使用的值可能取决于您对将浏览器与服务器隔开的网络中介的控制程度。
长轮询可以显著减少接收低延迟信息更新所需的请求数量,尤其是在新信息以不规则间隔提供的情况下。但是,更新越频繁,它就越接近传统轮询。
HTTP 流
浏览器向服务器发送请求,服务器在有信息要发送时响应。但是,与长轮询不同的是,服务器保持响应打开,并在新更新到达时继续发送更多更新。这种方法消除了轮询的需要,但也与典型的 HTTP 请求-响应语义有更大的偏离。例如,客户端和服务器需要就如何解释响应流达成一致,以便客户端知道一个更新在哪里结束,另一个更新在哪里开始。此外,网络中介可以缓存响应流,这会破坏这种方法的意图。这就是为什么今天长轮询更常用。
WebSocket 协议
浏览器向服务器发送 HTTP 请求以切换到 WebSocket 协议,服务器通过确认升级进行响应。此后,浏览器和服务器可以通过 TCP 套接字双向发送数据帧。
WebSocket 协议旨在取代轮询的需求,特别适用于需要在浏览器和服务器之间高频率交换消息的场景。通过 HTTP 的初始握手确保 WebSocket 请求能够通过防火墙。但是,也存在重大挑战,因为大多数已部署的浏览器不支持 WebSockets,并且在通过网络中介时存在更多问题。
WebSockets 围绕文本或二进制消息的双向交换展开。它导致了一种与基于 RESTful 的 HTTP 架构显著不同的方法。事实上,需要在 WebSockets 之上使用另一种协议,例如 XMPP、AMQP、STOMP 或其他协议,哪一个(或哪些)将成为主流还有待观察。
IETF 已经标准化了WebSocket 协议,而W3C 正在标准化的最后阶段中WebSocket API。已经出现了一些 Java 实现,包括 Jetty 和 Tomcat 等 servlet 容器。Servlet 3.1 规范可能会支持初始 WebSocket 升级请求,而单独的JSR-356 将定义基于 Java 的 WebSocket API。
回到 Spring MVC 3.2,Servlet 3 异步特性可用于长时间运行的请求以及 HTTP 流,Filip Hanik提到这些技术为“客户端 AJAX 调用的服务器版本”。至于 WebSockets,Spring 3.2 中尚不支持,但很可能会包含在 Spring 3.3 中。您可以关注SPR-9356 以获取进度更新。
下一篇文章 将转向示例代码,并更详细地解释新的 Spring MVC 3.2 功能。