上一次咱們實現WebSocket是基於Tomcat的方式,見上一篇博文《WebSocket實現方式一》javascript
用到的jar包以下:html
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-websocket</artifactId> <version>4.2.3.RELEASE</version> </dependency>
後臺代碼:前端
WebSocketConfig.java類java
/** * */ package nbs.omp.websocket; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Configuration; import org.springframework.stereotype.Component; import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter; import org.springframework.web.socket.config.annotation.EnableWebSocket; import org.springframework.web.socket.config.annotation.WebSocketConfigurer; import org.springframework.web.socket.config.annotation.WebSocketHandlerRegistry; import nbs.omp.interceptor.HandShakeInterceptor; /** * @author David * */ @Configuration @Component @EnableWebSocket public class WebSocketConfig extends WebMvcConfigurerAdapter implements WebSocketConfigurer { @Autowired MyWebSocketHandler handler; @Override public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) { registry.addHandler(handler, "/ws").addInterceptors(new HandShakeInterceptor()); } }
@Configuration //指明該類爲Spring 配置類web
@Component //泛指組件,當組件很差歸類的時候,咱們能夠使用這個註解進行標註。
@EnableWebSocket //聲明該類支持WebSocket
不要忘記在springmvc的配置文件中配置對此類的自動掃描spring
<!-- 自動掃描websocket包下的全部類 --> <context:component-scan base-package="nbs.omp.websocket" />
攔截器 HandShakeInterceptor.java類websocket
/** * */ package nbs.omp.interceptor; import java.util.Map; import javax.servlet.http.HttpSession; import org.springframework.http.server.ServerHttpRequest; import org.springframework.http.server.ServerHttpResponse; import org.springframework.http.server.ServletServerHttpRequest; import org.springframework.web.socket.WebSocketHandler; import org.springframework.web.socket.server.HandshakeInterceptor; import nbs.omp.model.User; /** * @author David * */ public class HandShakeInterceptor implements HandshakeInterceptor { @Override public boolean beforeHandshake(ServerHttpRequest request, ServerHttpResponse response, WebSocketHandler wsHandler, Map<String, Object> attributes) throws Exception { System.out.println("Websocket:用戶[ID:"+ ((ServletServerHttpRequest) request).getServletRequest() .getSession(false).getAttribute("user") + "]已經創建鏈接"); if (request instanceof ServletServerHttpRequest) { ServletServerHttpRequest servletRequest = (ServletServerHttpRequest) request; HttpSession session = servletRequest.getServletRequest().getSession(false); // 標記用戶 User user = (User) session.getAttribute("user"); if (user != null) { attributes.put("userId", user.getUserId()); } else { return false; } } return true; } @Override public void afterHandshake(ServerHttpRequest request, ServerHttpResponse response, WebSocketHandler wsHandler, Exception exception) { // TODO Auto-generated method stub } }
MyWebSocketHandler.java類session
/** * */ package nbs.omp.websocket; import java.io.IOException; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.Map.Entry; import java.util.concurrent.CopyOnWriteArrayList; import org.springframework.stereotype.Component; import org.springframework.web.socket.CloseStatus; import org.springframework.web.socket.TextMessage; import org.springframework.web.socket.WebSocketHandler; import org.springframework.web.socket.WebSocketMessage; import org.springframework.web.socket.WebSocketSession; /** * @author David * */ @Component public class MyWebSocketHandler implements WebSocketHandler { public static final List<Map<Integer, WebSocketSession>> userSocketSessionList; static { userSocketSessionList = new CopyOnWriteArrayList<Map<Integer, WebSocketSession>>(); } public void afterConnectionEstablished(WebSocketSession session) throws Exception { String userName = (String) session.getAttributes().get(""); //業務代碼... //發送消息 session.sendMessage(new TextMessage()); } @Override public void handleMessage(WebSocketSession session, WebSocketMessage<?> message) throws Exception { //業務代碼... //發送消息 sendMessageToUsers(new TextMessage()); } @Override public void handleTransportError(WebSocketSession session, Throwable exception) throws Exception { if (session.isOpen()) { session.close(); } for(Map userSocketSessionMap : userSocketSessionList){ Iterator<Entry<Integer, WebSocketSession>> it = userSocketSessionMap.entrySet().iterator(); // 移除Socket會話 while (it.hasNext()) { Entry<Integer, WebSocketSession> entry = it.next(); if (entry.getValue().getId().equals(session.getId())) { userSocketSessionList.remove(entry.getKey()); System.out.println("Socket會話已經移除:用戶ID" + entry.getKey()); break; } } } } @Override public void afterConnectionClosed(WebSocketSession session, CloseStatus closeStatus) throws Exception { System.out.println("Websocket:" + session.getId() + "已經關閉"); for(Map userSocketSessionMap : userSocketSessionList){ Iterator<Entry<Integer, WebSocketSession>> it = userSocketSessionMap.entrySet().iterator(); // 移除Socket會話 while (it.hasNext()) { Entry<Integer, WebSocketSession> entry = it.next(); if (entry.getValue().getId().equals(session.getId())) { userSocketSessionList.remove(userSocketSessionMap); System.out.println("Socket會話已經移除:用戶ID" + entry.getKey()); break; } } } } @Override public boolean supportsPartialMessages() { // TODO Auto-generated method stub return false; } /** * 給某個指定用戶發送消息 * * @param userName * @param message * @throws IOException */ public void sendMessageToUser(Integer userId, TextMessage message)throws IOException { for(Map userSocketSessionMap : userSocketSessionList){ WebSocketSession session = (WebSocketSession) userSocketSessionMap.get(userId); if (session != null && session.isOpen()) { session.sendMessage(message); } } } /** * 給全部在線用戶發送消息 * * @param message */ public void sendMessageToUsers(TextMessage message) { for (WebSocketSession user : users) { try { if (user.isOpen()) { user.sendMessage(message); } } catch (IOException e) { e.printStackTrace(); } } } }
前端JS:mvc
<script type="text/javascript"> var websocket; if ('WebSocket' in window) { websocket = new WebSocket("ws://" + path + "/ws"); } else if ('MozWebSocket' in window) { websocket = new MozWebSocket("ws://" + path + "/ws"); } else { websocket = new SockJS("http://" + path + "/ws/sockjs"); } websocket.onopen = function(event) { console.log("WebSocket:已鏈接"); console.log(event); }; websocket.onmessage = function(event) { var data=JSON.parse(event.data); console.log("WebSocket:收到一條消息",data); var textCss=data.from==-1?"sfmsg_text":"fmsg_text"; $("#content").append("<div class='fmsg'><label class='name'>"+data.fromName+" "+data.date+"</label><div class='"+textCss+"'>"+data.text+"</div></div>"); scrollToBottom(); }; websocket.onerror = function(event) { console.log("WebSocket:發生錯誤 "); console.log(event); }; websocket.onclose = function(event) { console.log("WebSocket:已關閉"); console.log(event); } function sendMsg(){ var data={}; websocket.send(JSON.stringify(data)); } } function send(event){ var code; if(window.event){ code = window.event.keyCode; // IE }else{ code = e.which; // Firefox } if(code==13){ sendMsg(); } } function clearAll(){ $("#content").empty(); } </script>
HTML頁面:app
<html> <body> <h1>Spring整合WebSocket實現</h1> <div id="content"></div> <input type="text" placeholder="請輸入要發送的信息" id="msg" class="msg" onkeydown="send(event)"> <input type="button" value="發送" class="send" onclick="sendMsg()" > <input type="button" value="清空" class="clear" onclick="clearAll()"> <div id="message"></div> </body> </html>
1,當spring容器啓動時會自動加載WebSocketConfig類,由於該類實現了WebSocketConfigurer接口,因此會調用registerWebSocketHandlers(...)方法。
2,當用戶登陸後頁面會調用JS執行websocket = new WebSocket("ws://" + path + "/ws");這行代碼,這時後臺攔截器會對「ws://" + path + "/ws」進行攔截,就進入了beforeHandshake(...)方法,握手成成功後進入afterHandshake(...)方法。
3,當攔截器順利經過後,程序就進入了MyWebSocketHandler類的supportsPartialMessages(...)方法,而後就是重要的afterConnectionEstablished(...)方法。連接成功會調用前端JS的websocket.onopen方法。
4,在afterConnectionEstablished(...)方法中就能夠寫業務邏輯並利用sendMessageToUser(...)發送消息了。
5,sendMessageToUser(...)方法的實質是利用webSocketSession.sendMessage(...)方法發送消息。
6,消息發送成功,這是會調用前端JS的websocket.onmessage方法。
7,當用戶經過頁面發送消息,即調用JS代碼websocket.send(JSON.stringify(data)),這時就進入了MyWebSocketHandler類的handleMessage(...)方法處理信息。
8,當有來自底層WebSocket消息傳輸錯誤時會調用後臺MyWebSocketHandler類的handleTransportError(...)方法進行WebSocketSession的close()等處理。
9,當用戶關閉WebSocket時會調用後臺MyWebSocketHandler類的afterConnectionClosed(...)方法移除會話。