介绍

现很多网站为了实现即时通讯,所用的技术都是轮询(polling)。轮询是在特定的的时间间隔(如每1秒),由浏览器对服务器发出HTTP request,然后由服务器返回最新的数据给客服端的浏览器。这种传统的HTTP request 的模式带来很明显的缺点 – 浏览器需要不断的向服务器发出请求,然而HTTP request 的header是非常长的,里面包含的数据可能只是一个很小的值,这样会占用很多的带宽。

而最比较新的技术去做轮询的效果是Comet – 用了AJAX。但这种技术虽然可达到全双工通信,但依然需要发出请求。

在 WebSocket API,浏览器和服务器只需要要做一个握手的动作,然后,浏览器和服务器之间就形成了一条快速通道。两者之间就直接可以数据互相传送。

依赖:

Tomcat 7 或者 J2EE7

  1. <dependency>
  2. <groupId>org.apache.tomcat</groupId>
  3. <artifactId>tomcat-websocket-api</artifactId>
  4. <version>7.0.47</version>
  5. <scope>provided</scope>
  6. </dependency>
  7.  
  8. <dependency>
  9. <groupId>javax</groupId>
  10. <artifactId>javaee-api</artifactId>
  11. <version>7.0</version>
  12. <scope>provided</scope>
  13. </dependency>

注意:早前业界没有统一的标准,各服务器都有各自的实现,现在J2EE7的JSR356已经定义了统一的标准,请尽量使用支持最新通用标准的服务器。

详见:http://www.oracle.com/technetwork/articles/java/jsr356-1937161.html

http://jinnianshilongnian.iteye.com/blog/1909962

我是用的Tomcat 7.0.57 + Java7

必须是Tomcat 7.0.47以上

详见:http://www.iteye.com/news/28414

ps:最早我们是用的Tomcat 7自带的实现,后来要升级Tomcat 8,结果原来的实现方式在Tomcat 8不支持了,就只好切换到支持Websocket 1.0版本的Tomcat了。

主流的java web服务器都有支持JSR365标准的版本了,请自行Google。

用nginx做反向代理的需要注意啦,socket请求需要做特殊配置的,切记!

Tomcat的处理方式建议修改为NIO的方式,同时修改连接数到合适的参数,请自行Google!

服务端

服务端不需要在web.xml中做额外的配置,Tomcat启动后就可以直接连接了。

  1. import com.dooioo.websocket.utils.SessionUtils;
  2. import org.apache.commons.logging.Log;
  3. import org.apache.commons.logging.LogFactory;
  4.  
  5. import javax.websocket.*;
  6. import javax.websocket.server.PathParam;
  7. import javax.websocket.server.ServerEndpoint;
  8.  
  9. /**
  10.  * 功能说明:websocket处理类, 使用J2EE7的标准
  11.  * 切忌直接在该连接处理类中加入业务处理代码
  12.  * 作者:liuxing(2014-11-14 04:20)
  13. */
  14. //relationId和userCode是我的业务标识参数,websocket.ws是连接的路径,可以自行定义
  15. @ServerEndpoint("/websocket.ws/{relationId}/{userCode}")
  16. public class WebsocketEndPoint {
  17.  
  18.  private static Log log = LogFactory.getLog(WebsocketEndPoint.class);
  19.  
  20. /**
  21.  * 打开连接时触发
  22.  * @param relationId
  23.  * @param userCode
  24.  * @param session
  25. */
  26. @OnOpen
  27.  public void onOpen(@PathParam("relationId") String relationId,
  28.  @PathParam("userCode") int userCode,
  29.  Session session){
  30.  log.info("Websocket Start Connecting:"+ SessionUtils.getKey(relationId, userCode));
  31.  SessionUtils.put(relationId, userCode, session);
  32. }
  33.  
  34. /**
  35.  * 收到客户端消息时触发
  36.  * @param relationId
  37.  * @param userCode
  38.  * @param message
  39.  * @return
  40. */
  41. @OnMessage
  42.  public String onMessage(@PathParam("relationId") String relationId,
  43.  @PathParam("userCode") int userCode,
  44.  String message) {
  45.  return"Got your message ("+ message +").Thanks !";
  46. }
  47.  
  48. /**
  49.  * 异常时触发
  50.  * @param relationId
  51.  * @param userCode
  52.  * @param session
  53. */
  54. @OnError
  55.  public void onError(@PathParam("relationId") String relationId,
  56.  @PathParam("userCode") int userCode,
  57.  Throwable throwable,
  58.  Session session) {
  59.  log.info("Websocket Connection Exception:"+ SessionUtils.getKey(relationId, userCode));
  60.  log.info(throwable.getMessage(), throwable);
  61.  SessionUtils.remove(relationId, userCode);
  62. }
  63.  
  64. /**
  65.  * 关闭连接时触发
  66.  * @param relationId
  67.  * @param userCode
  68.  * @param session
  69. */
  70. @OnClose
  71.  public void onClose(@PathParam("relationId") String relationId,
  72.  @PathParam("userCode") int userCode,
  73.  Session session) {
  74.  log.info("Websocket Close Connection:"+ SessionUtils.getKey(relationId, userCode));
  75.  SessionUtils.remove(relationId, userCode);
  76. }
  77.  
  78. }

工具类用来存储唯一key和连接

这个是我业务的需要,我的业务是服务器有对应动作触发时,推送数据到客户端,没有接收客户端数据的操作。

  1. import javax.websocket.Session;
  2. import java.util.Map;
  3. import java.util.concurrent.ConcurrentHashMap;
  4.  
  5. /**
  6.  * 功能说明:用来存储业务定义的sessionId和连接的对应关系
  7.  * 利用业务逻辑中组装的sessionId获取有效连接后进行后续操作
  8.  * 作者:liuxing(2014-12-26 02:32)
  9. */
  10. public class SessionUtils {
  11.  
  12.  public static Map<String, Session> clients = new ConcurrentHashMap<>();
  13.  
  14.  public static void put(String relationId, int userCode, Session session){
  15.  clients.put(getKey(relationId, userCode), session);
  16. }
  17.  
  18.  public static Session get(String relationId, int userCode){
  19.  return clients.get(getKey(relationId, userCode));
  20. }
  21.  
  22.  public static void remove(String relationId, int userCode){
  23.  clients.remove(getKey(relationId, userCode));
  24. }
  25.  
  26. /**
  27.  * 判断是否有连接
  28.  * @param relationId
  29.  * @param userCode
  30.  * @return
  31. */
  32.  public static boolean hasConnection(String relationId, int userCode) {
  33.  return clients.containsKey(getKey(relationId, userCode));
  34. }
  35.  
  36. /**
  37.  * 组装唯一识别的key
  38.  * @param relationId
  39.  * @param userCode
  40.  * @return
  41. */
  42.  public static String getKey(String relationId, int userCode) {
  43.  return relationId +"_"+ userCode;
  44. }
  45.  
  46. }

推送数据到客户端

在其他业务方法中调用

  1. /**
  2.  * 将数据传回客户端
  3.  * 异步的方式
  4.  * @param relationId
  5.  * @param userCode
  6.  * @param message
  7. */
  8.  public void broadcast(String relationId, int userCode, String message) {
  9.  if (TelSocketSessionUtils.hasConnection(relationId, userCode)) {
  10.  TelSocketSessionUtils.get(relationId, userCode).getAsyncRemote().sendText(message);
  11.  } else {
  12.  throw new NullPointerException(TelSocketSessionUtils.getKey(relationId, userCode) +"Connection does not exist");
  13. }
  14.  
  15. }

我是使用异步的方法推送数据,还有同步的方法

详见:http://docs.oracle.com/javaee/7/api/javax/websocket/Session.html

客户端

  1. var webSocket = null;
  2. var tryTime = 0;
  3. $(function () {
  4. initSocket();
  5.  
  6.  window.onbeforeunload = function () {
  7. //离开页面时的其他操作
  8. };
  9. });
  10.  
  11. /**
  12.  * 初始化websocket,建立连接
  13. */
  14. function initSocket() {
  15.  if (!window.WebSocket) {
  16. alert("您的浏览器不支持websocket!");
  17.  return false;
  18. }
  19.  
  20.  webSocket = new WebSocket("ws://127.0.0.1:8080/websocket.ws/"+ relationId +"/"+ userCode);
  21.  
  22.  // 收到服务端消息
  23.  webSocket.onmessage = function (msg) {
  24. console.log(msg);
  25. };
  26.  
  27.  // 异常
  28.  webSocket.onerror = function (event) {
  29. console.log(event);
  30. };
  31.  
  32.  // 建立连接
  33.  webSocket.onopen = function (event) {
  34. console.log(event);
  35. };
  36.  
  37.  // 断线重连
  38.  webSocket.onclose = function () {
  39.  // 重试10次,每次之间间隔10秒
  40.  if (tryTime < 10) {
  41.  setTimeout(function () {
  42.  webSocket = null;
  43. tryTime++;
  44. initSocket();
  45.  }, 500);
  46.  } else {
  47.  tryTime = 0;
  48. }
  49. };
  50.  
  51. }

其他调试工具

Java实现一个websocket的客户端

  1. <dependency>
  2. <groupId>org.java-websocket</groupId>
  3. <artifactId>Java-WebSocket</artifactId>
  4. <version>1.3.0</version>
  5. </dependency>
  1. import java.io.IOException; 
  2.  import javax.websocket.ClientEndpoint; 
  3.  import javax.websocket.OnError; 
  4.  import javax.websocket.OnMessage; 
  5.  import javax.websocket.OnOpen; 
  6.  import javax.websocket.Session; 
  7.  
  8. @ClientEndpoint
  9.  public class MyClient { 
  10. @OnOpen
  11.  public void onOpen(Session session) { 
  12.  System.out.println("Connected to endpoint:"+ session.getBasicRemote()); 
  13.  try { 
  14. session.getBasicRemote().sendText("Hello");
  15.  } catch (IOException ex) { 
  16. }
  17. }
  18.  
  19. @OnMessage
  20.  public void onMessage(String message) { 
  21. System.out.println(message);
  22. }
  23.  
  24. @OnError
  25.  public void onError(Throwable t) { 
  26. t.printStackTrace();
  27. }
  28. }
  1. import java.io.BufferedReader; 
  2.  import java.io.IOException; 
  3.  import java.io.InputStreamReader; 
  4.  import java.net.URI; 
  5.  import javax.websocket.ContainerProvider; 
  6.  import javax.websocket.DeploymentException; 
  7.  import javax.websocket.Session; 
  8.  import javax.websocket.WebSocketContainer; 
  9.  
  10.  public class MyClientApp { 
  11.  
  12.  public Session session; 
  13.  
  14.  protected void start() 
  15. {
  16.  
  17.  WebSocketContainer container = ContainerProvider.getWebSocketContainer(); 
  18.  
  19.  String uri ="ws://127.0.0.1:8080/websocket.ws/relationId/12345"; 
  20.  System.out.println("Connecting to"+ uri); 
  21.  try { 
  22.  session = container.connectToServer(MyClient.class, URI.create(uri)); 
  23.  } catch (DeploymentException e) { 
  24. e.printStackTrace();
  25.  } catch (IOException e) { 
  26. e.printStackTrace();
  27. }
  28.  
  29. }
  30.  public static void main(String args[]){ 
  31.  MyClientApp client = new MyClientApp(); 
  32. client.start();
  33.  
  34.  BufferedReader br = new BufferedReader(new InputStreamReader(System.in)); 
  35.  String input =""; 
  36.  try { 
  37. do{
  38.  input = br.readLine(); 
  39. if(!input.equals("exit"))
  40. client.session.getBasicRemote().sendText(input);
  41.  
  42. }while(!input.equals("exit"));
  43.  
  44.  } catch (IOException e) { 
  45.  // TODO Auto-generated catch block 
  46. e.printStackTrace();
  47. }
  48. }
  49. }

Chrome安装一个websocket模拟客户端

最后

为了统一的操作体验,对于一些不支持websocket的浏览器,请使用socketjs技术做客户端开发。

websocket 实现消息推送(转)的更多相关文章

  1. 在Spring Boot框架下使用WebSocket实现消息推送

    Spring Boot的学习持续进行中.前面两篇博客我们介绍了如何使用Spring Boot容器搭建Web项目(使用Spring Boot开发Web项目)以及怎样为我们的Project添加HTTPS的 ...

  2. WebSocket与消息推送

    B/S结构的软件项目中有时客户端需要实时的获得服务器消息,但默认HTTP协议只支持请求响应模式,这样做可以简化Web服务器,减少服务器的负担,加快响应速度,因为服务器不需要与客户端长时间建立一个通信链 ...

  3. HTML5 学习总结(五)——WebSocket与消息推送

    B/S结构的软件项目中有时客户端需要实时的获得服务器消息,但默认HTTP协议只支持请求响应模式,这样做可以简化Web服务器,减少服务器的负担,加快响应速度,因为服务器不需要与客户端长时间建立一个通信链 ...

  4. HTML5 学习笔记(五)——WebSocket与消息推送

    B/S结构的软件项目中有时客户端需要实时的获得服务器消息,但默认HTTP协议只支持请求响应模式,这样做可以简化Web服务器,减少服务器的负担,加快响应速度,因为服务器不需要与客户端长时间建立一个通信链 ...

  5. 使用websocket进行消息推送服务

    Websocket主要做消息推送,简单,轻巧,比comet好用 入门了解:https://www.cnblogs.com/xdp-gacl/p/5193279.html /** * A Web Soc ...

  6. 实现websocket 主动消息推送,用laravel+Swoole

    近来有个需求:想实现一个可以主动触发消息推送的功能,这个可以实现向模板消息那个,给予所有成员发送自定义消息,而不需要通过客户端发送消息,服务端上message中监听传送的消息进行做相对于的业务逻辑. ...

  7. 【转】SpringMVC整合websocket实现消息推送及触发

    1.创建websocket握手协议的后台 (1)HandShake的实现类 /** *Project Name: price *File Name:    HandShake.java *Packag ...

  8. 用图解&&实例讲解php是如何实现websocket实时消息推送的

    WebSocket是 HTML5 开始提供的一种在单个 TCP 连接上进行全双工通讯的协议. 以前的推送技术使用 Ajax 轮询,浏览器需要不断地向服务器发送http请求来获取最新的数据,浪费很多的带 ...

  9. WebSocket 学习教程(二):Spring websocket实现消息推送

    =============================================== 环境介绍: Jdk 1.7 (1.6不支持) Tomcat7.0.52 (支持Websocket协议) ...

随机推荐

  1. 云栖干货回顾 | 云原生数据库POLARDB专场“硬核”解析

    POLARDB是阿里巴巴自主研发的云原生关系型数据库,目前兼容三种数据库引擎:MySQL.PostgreSQL.Oracle.POLARDB的计算能力最高可扩展至1000核以上,存储容量可达100TB ...

  2. 在myeclipse安装beyond插件

    目录 文章背景 目录 环境介绍 安装步骤 说明 参考文章 版本记录 文章背景 myeclipse自带的比较工具感觉是有一些看不清晰,也不是太方便处理,然后就找了个比较插件了. 目录 环境介绍 myec ...

  3. hadoop的基准测试

    hadoop的基准测试 实际生产环境当中,hadoop的环境搭建完成之后,第一件事情就是进行压力测试,测试我们的集群的读取和写入速度,测试我们的网络带宽是否足够等一些基准测试 测试写入速度 向HDFS ...

  4. NX二次开发-UFUN工程图导入视图UF_DRAW_import_view

    NX9+VS2012 #include <uf.h> #include <uf_draw.h> #include <uf_obj.h> #include <u ...

  5. NX二次开发-NXOpen获取边的端点NXOpen::Edge::GetVertices

    NX9+VS2012 #include <NXOpen/Features_BlockFeatureBuilder.hxx> #include <NXOpen/Features_Fea ...

  6. Python 爬虫-抓取小说《鬼吹灯之精绝古城》

    想看小说<鬼吹灯之精绝古城>,可是网页版的好多广告,还要一页一页的翻,还无法复制,于是写了个小爬虫,保存到word里慢慢看. 代码如下: """ 爬取< ...

  7. Function Run Fun-递归+细节处理

    We all love recursion! Don't we? Consider a three-parameter recursive function w(a, b, c): if a < ...

  8. C#委托的实质

    1,委托时方法指针: 2,委托时一个类,对其进行实例化的时候,要将引用的方法作为他的构造方法的参数.

  9. IDEA @Autowired dao大红波浪线

    SptingBoot+Mybatis开发通常在dao层的注解是@Mapper 这样每次在ServiceImpl层加注解@Autowired时,注入的dao总是波浪线烦人,其实并没有错,只是idea你太 ...

  10. 第一周课堂笔记1th

    1.      计算机基本组成 CPU:中央处理器,人的大脑 内存:临时存储数据       (特点:断电即消失,容量小12G 24G) 硬盘:永久存储数据(特点:断电不消失容量大,500G 1T) ...