标签:img 处理 ror 思考 延时 while scope 间隔 缺点
首先要知道为什么使用服务器推送,回答这个问题其实就是相当于回答,服务器推送的优点,可以从两个方面来思考:
及时的将客户端感兴趣的数据推送给它。
不使用服务端推送,那就只能由客户端定期对服务器发送请求,来获取是否有需要的数据。这样做有几个缺点:
我理解的有两个好处,一是及时,还有就是消耗资源稳定(消耗一个连接数)。及时很好理解,就是服务器知道数据什么时候发生变化,发生变化的时候就进行推送。而消耗资源稳定,则是因为只有一个连接,所有的数据都从这么连接发送。
当然就是每一个客户端都需要维护一个长连接,客户端数量增多的时候,会对服务器造成较大的压力。
如果满足以下条件,那么使用它是最好的做法,若不全部满足,则酌情考虑:
服务器推送的实现有很多种方式,这一篇博客使用EventSource来实现功能。若要实现服务器推送,需要客户端和服务端同时对其进行支持。
客户端代码比较简单,实现一个回调函数即可:
new EventSource("longConnection").onmessage = function(event) { $scope.$apply(function() { alert(event.data); }); };
回调函数中的event.data就是服务器发送的数据,此时可以对它进行其它操作。
服务端的代码一般放到一个循环中:
response.setContentType("text/event-stream;charset=UTF-8"); response.setHeader("Cache-Control","no-cache"); response.setHeader("Connection","keep-alive"); PrintWriter out=response.getWriter(); while(true){ out.print("data: " + 传递的数据 + "\n\n"); out.flush(); }
只要不把连接关闭,那么每一次刷新,都会讲数据发送到客户端,并触发onmessage方法。
默认IE是不支持EventSource对象的,解决办法是引入一个js文件eventsource.min.js,但是也只能支持IE8及以上。
从上面的例子中可以发现,往客户端发送数据的代码是写在一个死循环中,那么怎么才能实现当敏感数据发生变化时,才使其执行呢,可行性办法有很多,现在提供一个方法,在全局范围内使用LinkedBlockingQueue对象,当有需要发送的数据时,将数据放到队列中,然后在循环中调用poll方法即可。
如果细心的话可以发现,在服务端代码中,写数据的使用使用了如下格式:
out.print("data: " + 传递的数据 + "\n\n");
out.flush();
每次写完刷新可不用说,重点是写数据的时候,有一个前缀和后缀,那么这个是有什么规定吗,还是说与页面js代码一一对应即可,答案是有规定的。下面使用F12对长连接进行观察,可以看到如下内容:
具体可以使用哪些前缀,有id,data,event,retry和空白,而后缀使用两个换行,实际上这代表着是一个空行,学习过servlet上传的可能会理解。
一般用法都是在传递的数据中使用json,里面存放需要调用的方法名,调用参数放在后面,这样就可以根据不同的数据类型进行不同的处理。
同样的客户端代码中使用了onmessage方法,那么是否只有这一个方法呢,其实还有额外的两个方法,这两个方法视情况使用 :
这个只是服务端推送的一种实现方式,基本能满足要求,还有其他方式,后面会介绍。
标签:img 处理 ror 思考 延时 while scope 间隔 缺点
原文地址:http://www.cnblogs.com/lzzkz-lxy/p/6416384.html