hello~亲爱的看官老爷们大家好~过完年第一周已经结束,是时候开始制定新的工作计划了。主要负责的项目是数据可视化平台,而使用中如果服务器能有推送能力让页端得到相关通知的话,就能实现很多功能上的优化。鉴于项目中 Node 端已经正式投入使用,前端拥有了自己的服务器,搞事情起来自然方便很多。

技术选型:SSE(Server-sent Events) or WebSocket

若干年前,服务器并没有主动推送的能力,主要是通过轮询的方式来达到近似于服务器推送的能力。现在不需要这么麻烦,轮询只作为向下兼容的方案即可,当前主流的服务器推送是使用 SSE 或者 WebSocket 来实现的。两者对比如下:

是否基于新协议 是否双向通信 是否支持跨域 接入成本
SSE 否(Http 否(服务器单向) 否(Firefox 支持跨域)
WebSocket 是(ws

需要稍微解释一下的是接入成本。SSE 是相对轻量级的协议,(Node)代码实现上比较简单,而 WebSocket 是比较复杂的协议,虽然也有类库可以套用,也许页端方面两者代码量差不多,但服务器方面实现就复杂不少了。同时,要实现 WebSocket,是需要另起一个服务,而 SSE 并不需要。

比较之后,对 SSE 与 WebSocket 有了大致的理解。项目对服务器推送的要求是发送通知,而未来可能需要接入实时同步的功能,结合项目实际情况与接入成本后,选择了 SSE。

最后看一下浏览器支持情况以作参考:


IElet it go吧,日常不支持~其他浏览器还是绿油油的,支持度还是挺高的。

示例

Node端

项目中使用 Egg 作为框架,底层是 Koa2 的,因而使用 Koa2 作为示例。Node 端关键代码如下:

  1. app.use(async (ctx) => {
  2. const { res, request: { url } } = ctx;
  3. res.writeHead(200, {
  4. 'Content-Type': 'text/event-stream', // 服务器声明接下来发送的是事件流
  5. });
  6. let stream = new PassThrough();
  7. let i = 0;
  8. let timer = setInterval(() => {
  9. if (i === 5) {
  10. stream.write('event: pause\n'); // 事件类型
  11. } else {
  12. stream.write('event: test\n'); // 事件类型
  13. }
  14. stream.write(`id: ${+new Date()}\n`); // 消息ID
  15. stream.write(`data: ${i}\n`); // 消息数据
  16. stream.write('retry: 10000\n'); // 重连时间
  17. stream.write('\n\n'); // 消息结束
  18. i++;
  19. }, 1000);
  20. stream.on('close', function() {
  21. console.log('closed.')
  22. clearInterval(timer);
  23. })
  24. ctx.body = stream;
  25. });

服务器告诉客户端,返回的类型是事件流(text/event-stream),查阅 MDN 文档可知:事件流仅仅是一个简单的文本数据流,文本应该使用 UTF- 8 格式的编码。每条消息后面都由一个空行作为分隔符。以冒号开头的行为注释行,会被忽略。

之后就是消息主体了,尽管例子使用 setInterval 模拟不断发送推送,但换成任意条件触发推送也是可以的。stream.write 调用了5次,对应规范中的各个字段,理解如下:

  • event 为消息的事件类型。客户端在 EventSource 中可以通过 addEventListener 收听相关的消息。该字段可省略,省略后客户端触发 message 事件。
  • id 为事件 ID。作为客户端内部的“最后一个事件 ID”的属性值,用于重连,不可省略。
  • data 为消息的数据字段,简单说就是客户端监听时间后,通过e.data 拿到的数据。
  • retry 为重连时间,可省略该参数。
  • 最后是结束该次通知的 \n\n,不可省略。除了上面规定的字段名,其他所有的字段名都会被忽略。

更详细的解释可以查阅 MDN 文档。有一个小细节需要注意,在 SSE 的草案中提到,"text/event-stream" 的 MIME 类型传输应当在静置 15 秒后自动断开。但实测(仅用了 Chrome)后发现,即使静置时间超过 15 秒,浏览器与客户端均不会断开连接。查阅了不少文章,均建议维护一套发送 \n\n 的心跳机制。个人认为此举有助于提高客户端程序的健壮性,但不是必须的。

最后是监听事件流的 close 事件,用于结束此次的链接。测试后发现,无论是让客户端调用 close 方法(下文有例子~),还是异常结束,包括关闭窗口、关闭程序等,都能触发服务器的 close 事件。

客户端

客户端代码更简单,示例如下:

  1. const source = new EventSource('http://localhost:3000/test');
  2. source.addEventListener('open', () => {
  3. console.log('Connected');
  4. }, false);
  5. source.addEventListener('message', e => {
  6. console.log(e.data);
  7. }, false);
  8. source.addEventListener('pause', e => {
  9. source.close();
  10. }, false);

前端童鞋对于这样的代码应该挺熟悉的,一切都是事件触发,根据不同的事件执行对应的代码。稍微说明一下 EventSource 拥有的属性和方法,相信大家就可以愉快地使用了。

EventSource 有三个默认的事件,分别是:

  • open:在连接打开时被调用。
  • message:收到一个没有 event 属性的消息时被调用。
  • error:当发生错误时被调用。

两个只读属性:

  • readyState:代表连接状态。可能值是CONNECTING (0), OPEN (1), 或者 CLOSED (2)。
  • url:代表连接的 URL。

一个方法:

  • close:调用后关闭连接(也就是上文所提及的)。

更详细的解释可以查阅 MDN 文档

小结

关于服务器 SSE 的简单介绍就到此为止了,可以看到,SSE 开发起来还是比较简单的,接入成本非常低。但并不是说 WebSocket 就是不好的,抛开实际场景谈业务就是耍流氓。此外上述代码只是演示,还能进一步进行优化的。如为了减轻服务器开销,可以建立一套机制有目的地断开与重连等,大家可以自行实现。

相关的代码已经丢到 Github 上,欢迎查阅。

感谢各位看官大人看到这里,知易行难,希望本文对你有所帮助~谢谢!

参考资料

20 行代码写一个数据推送服务

使用服务器发送事件

EventSource

服务器推送 SSE 了解一下?的更多相关文章

  1. SSE技术详解:一种全新的HTML5服务器推送事件技术

    前言 一般来说,Web端即时通讯技术因受限于浏览器的设计限制,一直以来实现起来并不容易,主流的Web端即时通讯方案大致有4种:传统Ajax短轮询.Comet技术.WebSocket技术.SSE(Ser ...

  2. 前端通信:SSE设计方案(二)--- 服务器推送技术的实践以及一些应用场景的demo(包括在线及时聊天系统以及线上缓存更新,代码热修复案例)

    距离上一篇博客,这篇文章的发布大概过了整整三个月.我也从饿了么度过了试用期,成为了正式员工.刚进来恰好遇到项目底层改造和迁移,将项目从angular全部迁移到vue上,所以适应这边的节奏和业务的开发任 ...

  3. HTML5中的SSE(服务器推送技术)

    本文原链接:https://cloud.tencent.com/developer/article/1194063 SSE技术详解:一种全新的HTML5服务器推送事件技术 前言 概述 基本介绍 与We ...

  4. SSE:服务器推送事件

    SSE:Server-Sent Event,服务器推送事件 常规的Http协议是一个请求对应一个响应的这种方式的 但对于某些实时性要求比较高的需求,HTML5中新增了SSE,可以很方便的实现局部数据的 ...

  5. HTML5 服务器推送事件(Server-sent Events)实战开发

    转自:http://www.ibm.com/developerworks/cn/web/1307_chengfu_serversentevent/ http://www.ibm.com/develop ...

  6. JavaEE开发之SpringMVC中的静态资源映射及服务器推送技术

    在上篇博客中,我们聊了<JavaEE开发之SpringMVC中的自定义拦截器及异常处理>.本篇博客我们继续的来聊SpringMVC的东西,下方我们将会聊到js.css这些静态文件的加载配置 ...

  7. HTML5 服务器推送事件(Server-sent Events)

    服务器推送事件(Server-sent Events)WebSocket 协议的一种服务器向客户端发送事件&数据的单向通讯.目前所有主流浏览器均支持服务器发送事件,当然除了 Internet ...

  8. 【SpringBoot】服务器端主动推送SSE技术讲解

    =====================16.高级篇幅之SpringBoot2.0服务器端主动推送SSE技术讲解 ============================ 1.服务端推送常用技术介绍 ...

  9. springweb flux 服务器推送事件

    以前做服务器推送一般用轮询,后端主动给客户端推送不是很好解决.有时候也可以采用websocket 现在看了springwebflux,用它自带的方法做服务器推送方便多了. 代码如下: import o ...

随机推荐

  1. 二十九、rsync+inotity实时监控同步工具

    一.场景应用:                                    客户通过url访问资源(查询,下载等),并发量是非常高的,所以运用负载均衡分担web服务器的压力,在后端连接不同的 ...

  2. vue实现动态绑定class--(boolean)绑定class,点击有,再点击取消

    <template> <div :class="{'flag':selected}" @click=clickBtn>xxx</div>< ...

  3. 1、简述在java网络编程中,服务端程序与客户端程序的具体开发步骤?

    网络编程分为UDP通信和TCP通信 UDP协议: 发送端:1.创建DatagramSocket对象.2.创建DatagramPacket对象,并封装数据.3.发送数据.4.释放 资源. 接收端:1.创 ...

  4. OneDrive for Business

    一.界面介绍 1.在Office 365主页 点击“OneDrive”登陆 2.进入OneDrive,可对文档进行存储.同步并共享文档. 3.点击,可对文档进行编辑.分享.重命名等操作 二.文档同步 ...

  5. 未来科技城 x 奇点云打造「企业数据大脑」,助力1.3万家企业服务

    “当前,政府数字化和数字政府建设已成为一种趋势.一种必然,并且有了一条水到渠成式的实现路径.” 上升为国家战略的数字中国建设加速了”智慧政务“的生动实践,杭州未来科技城的「企业数据大脑」就是一个典型. ...

  6. Leetcode13_罗马数字转整数

    题目 罗马数字包含以下七种字符: I, V, X, L,C,D 和 M. 字符 数值I 1V 5X 10L 50C 100D 500M 1000例如, 罗马数字 2 写做 II ,即为两个并列的 1. ...

  7. Session深入浅出

    Session会在浏览器关闭后消失吗? 通常情况下,当我们关闭浏览器再重新打开后,我们就需要再次进行登陆(如果没有进行下次自动登录之类的设置).在Jav中(Session是通用的,这里以Java为例) ...

  8. JavaScript学习总结(五)原型和原型链详解

    转自:http://segmentfault.com/a/1190000000662547 私有变量和函数 在函数内部定义的变量和函数,如果不对外提供接口,外部是无法访问到的,也就是该函数的私有的变量 ...

  9. Netflix拒上戛纳电影节,能给国内视频产业什么启示?

    当新事物诞生时,总是会遭到质疑,甚至是排斥!因为新事物的活力.潜力,都对保守的传统事物产生了极大的冲击.就像有声电影刚刚诞生时,一代"默片大师"卓别林就对其进行了激烈的反对.他认为 ...

  10. Mysql的分页查询语句的性能分析

    MySql分页sql语句,如果和MSSQL的TOP语法相比,那么MySQL的LIMIT语法要显得优雅了许多.使用它来分页是再自然不过的事情了. 1.1最基本的分页方式: 在中小数据量的情况下,这样的S ...