WebSocket整合SSM(Spring,Struts2,Maven)

1、WebSocket与HTTP长轮询html

  WebSocket 属于HTML5 规范的一部分,提供的一种在单个 TCP 链接上进行全双工通信的协议。容许服务端主动向客户端推送数据。在 WebSocket API 中,浏览器和服务器只须要完成一次握手,二者之间就直接能够建立持久性的链接,并进行双向数据传输。(Tomcat 8以上版本支持)java

  HTTP 协议是一种无状态的、无链接的、单向的应用层协议。它采用了请求/响应模型。通讯请求只能由客户端发起,服务端对请求作出应答处理。这种通讯模型有一个弊端:HTTP 协议没法实现服务器主动向客户端发起消息。web

  HTTP长轮询:客户端向服务器发送(http)Ajax请求,服务器接到请求后hold住链接,直到有新消息才返回响应信息并关闭链接,客户端处理完响应信息后再向服务器发送新的请求。spring

  WebSocket与HTTP长轮询的通讯机制以下:api

2、客户端如何使用WebSocket浏览器

  浏览器经过 JavaScript 向服务器发出创建 WebSocket 链接的请求,链接创建之后,客户端和服务器端就能够经过 TCP 链接直接交换数据。安全

  当你获取 Web Socket 链接后,你能够经过 send() 方法来向服务器发送数据,并经过 onmessage 事件来接收服务器返回的数据。服务器

  1. 建立 WebSocket 对象API

  //第一个参数 url, 指定链接的 URL。第二个参数 protocol 是可选的,指定了可接受的子协议。  微信

  API:var webSocket = new WebSocket(url, [protocol] );websocket

  1. WebSocket 属性 如下是 WebSocket 对象的属性。假定咱们使用了以上代码建立了 Socket 对象:

属性 描述 Socket.readyState 只读属性 readyState 表示链接状态,能够是如下值:

0 - 表示链接还没有创建。

1 - 表示链接已创建,能够进行通讯。

2 - 表示链接正在进行关闭。

3 - 表示链接已经关闭或者链接不能打开。

Socket.bufferedAmount 只读属性 bufferedAmount 已被 send() 放入正在队列中等待传输,可是尚未发出的 UTF-8 文本字节数。

  1. WebSocket 事件 如下是 WebSocket 对象的相关事件。假定咱们使用了以上代码建立了 Socket 对象:

事件 事件处理程序 描述 open Socket.onopen 链接创建时触发 message Socket.onmessage 客户端接收服务端数据时触发 error Socket.onerror 通讯发生错误时触发 close Socket.onclose 链接关闭时触发

  1. WebSocket 方法 如下是 WebSocket 对象的相关方法。假定咱们使用了以上代码建立了 Socket 对象:

方法 描述 Socket.send() 使用链接发送数据

Socket.close() 关闭链接

  1. WebSocket 实例 <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
Testing WebSocket
<textarea id="message" readonly></textarea>    <br />    
<input id="msg" type="text" />    
<button type="submit" id="sendButton" onClick="postToServer()">Send!</button>    
<button type="submit" id="sendButton" onClick="onClose()">End</button>
复制代码
3、服务器端如何使用WebSocket
  1. pom.xml文件引入对应的依赖包:

    <!-- webSocket 开始-->
     <dependency>
         <groupId>javax.websocket</groupId>
         <artifactId>javax.websocket-api</artifactId>
         <version>1.1</version>
         <scope>provided</scope>
     </dependency>
      
     <dependency>
         <groupId>javax</groupId>
         <artifactId>javaee-api</artifactId>
         <version>7.0</version>
         <scope>provided</scope>
     </dependency>
     <!-- webSocket 结束-->        
    复制代码
  2. spring配置文件配置websocket服务类

  3. struts2配置文件过滤websocket请求

4. 定义Websocket.java类

@ServerEndpoint("/ws/websocket") public class WebSocket { private static Logger logger = Logger.getLogger(WebSocket.class); @SuppressWarnings("static-access") private MessageHandler messageHandler = getWebSocketService().messageHandler;

//静态变量,用来记录当前在线链接数。应该把它设计成线程安全的
public static int onlineCount = 0;

//与某个客户端的链接会话,须要经过它来给客户端发送数据
public Session session;

/**
 * 链接创建成功调用的方法
 * @param session  可选的参数。session为与某个客户端的链接会话,须要经过它来给客户端发送数据
 */
@OnOpen
public void onOpen(Session session){
    this.session = session;
    WebSocketMapUtil.put(session.getId(),this);//存入map
    addOnlineCount();           //在线数加1
    messageHandler.onConnect(session.getId());
    System.out.println("有新链接加入!当前在线人数为" + getOnlineCount());
    logger.info("有新链接加入!当前在线人数为" + getOnlineCount());
}

/**
 * 链接关闭调用的方法
 */
@OnClose
public void onClose(){
    WebSocketMapUtil.remove(session.getId());  //从map中删除
    subOnlineCount();           //在线数减1
    messageHandler.onDisconnect(session.getId());
    try {
        if(session != null){
            session.close();
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
    System.out.println("有一链接关闭!当前在线人数为" + getOnlineCount());
    logger.info("有一链接关闭!当前在线人数为" + getOnlineCount());
}

/**
 * 收到客户端消息后调用的方法
 * @param message 客户端发送过来的消息
 * @param session 可选的参数
 */
@OnMessage
public String onMessage(String message, Session session) {
    System.out.println("来自客户端的消息:" + message);
    logger.info("来自客户端的消息:" + message);
    if (message.equals("&")){
        return "&";
    }else{
         messageHandler.onMessage(session.getId(),message);
        return "Got your message ("+ message +")";
    }
}

/**
 * 发生错误时调用
 * @param session
 * @param error
 */
@OnError
public void onError(Session session, Throwable error){
    System.out.println("发生错误");
    logger.info("发生错误:" + error);
    error.printStackTrace();
}
复制代码

//单发消息 public void sendMessage(String message) throws IOException{ //阻塞式(同步) //this.session.getBasicRemote().sendText(message); //非阻塞式(异步) this.session.getAsyncRemote().sendText(message); }

//群发消息
public void sendMessageAll(String message) throws IOException{
    for(WebSocket webSocket : WebSocketMapUtil.getValues()){
        webSocket.sendMessage(message);
    }
}

public static synchronized int getOnlineCount() {
    return onlineCount;
}

public static synchronized void addOnlineCount() {
    WebSocket.onlineCount++;
}

public static synchronized void subOnlineCount() {
    WebSocket.onlineCount--;
}

public WebSocketServiceImpl getWebSocketService(){
    ClassPathXmlApplicationContext classPathXmlApplicationContext = new ClassPathXmlApplicationContext("classpath*:applicationContext.xml");
    return (WebSocketServiceImpl) classPathXmlApplicationContext.getBean("webSocketService");
}
复制代码

} 5. 在WebSocketServiceImpl实现类里实现服务器推送消息的方法

@Override public boolean sendMessage(String message) { logger.info("发送给客户端的websocket消息message=="+message); if (messageHandler == null) { logger.info("上层调用者没有给messageHandler赋值,websocket消息没法发送到客户端"); return false; } if (message == null || "".equals(message)) { logger.info("发送给客户端的消息message为空"); return false; } try { for(WebSocket webSocket : WebSocketMapUtil.getValues()){ webSocket.sendMessage(message); logger.info("给sessionId为" + webSocket.session.getId() + "的客户端发送消息:"+message+"=成功"); }
return true; } catch (IOException e) { logger.info("向全部客户端发送消息:"+message+"=失败:" + e); e.printStackTrace(); } return false; } 其实,使用Tomcat8开发WebSocket服务端很是简单,主要有以下两种方式:

  1. 使用注解方式开发(上述示例代码),被@ServerEndpoint修饰的Java类便可做为WebSocket服务端。

  2. 继承Endpint基类实现WebSocket服务端。

如需转载,请注明做者(冰湖一角)和连接https://www.cnblogs.com/bingyimeiling/p/10276801.html

欢迎关注微信公众号【Java典籍】,收看更多Java技术干货!

 ▼微信扫一扫下图↓↓↓二维码关注

相关文章
相关标签/搜索