Java中Websocket使用例項解讀
介紹
現在很多網站為了實現即時通訊,所用的技術都是輪詢(polling)。輪詢是在特定的的時間間隔(如每1秒),由瀏覽器對伺服器發出HTTP request,然後由伺服器返回最新的資料給客服端的瀏覽器。
這種傳統的HTTP request 的模式帶來很明顯的缺點 – 瀏覽器需要不斷的向伺服器發出請求,然而HTTP request 的header是非常長的,裡面包含的資料可能只是一個很小的值,這樣會佔用很多的頻寬。
而最比較新的技術去做輪詢的效果是comet – 用了AJAX。但這種技術雖然可達到全雙工通訊,但依然需要發出請求。
在 WebSocket API,瀏覽器和伺服器只需要要做一個握手的動作,然後,瀏覽器和伺服器之間就形成了一條快速通道。兩者之間就直接可以資料互相傳送。
執行環境:
客戶端
實現了websocket的瀏覽器
Chrome | Supported in version 4+ |
Firefox | Supported in version 4+ |
Internet Explorer | Supported in version 10+ |
Opera | Supported in version 10+ |
Safari | Supported in version 5+ |
服務端
依賴
Tomcat 7.0.47以上 + J2EE7
<dependency> <groupId>org.apache.tomcat</groupId> <artifactId>tomcat-websocket-api</artifactId> <version>7.0.47</version> <scope>provided</scope> </dependency> <dependency> <groupId>javax</groupId> <artifactId>javaee-api</artifactId> <version>7.0</version> <scope>provided</scope> </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啟動後就可以直接連線了。
實現
import com.dooioo.websocket.utils.SessionUtils; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import javax.websocket.*; import javax.websocket.server.PathParam; import javax.websocket.server.ServerEndpoint; /** * 功能說明:websocket處理類, 使用J2EE7的標準 * 切忌直接在該連線處理類中加入業務處理程式碼 * 作者:liuxing(2014-11-14 04:20) */ //relationId和userCode是我的業務標識引數,websocket.ws是連線的路徑,可以自行定義 @ServerEndpoint("/websocket.ws/{relationId}/{userCode}") public class WebsocketEndPoint { private static Log log = LogFactory.getLog(WebsocketEndPoint.class); /** * 開啟連線時觸發 * @param relationId * @param userCode * @param session */ @OnOpen public void onOpen(@PathParam("relationId") String relationId, @PathParam("userCode") int userCode, Session session){ log.info("Websocket Start Connecting: " + SessionUtils.getKey(relationId, userCode)); SessionUtils.put(relationId, userCode, session); } /** * 收到客戶端訊息時觸發 * @param relationId * @param userCode * @param message * @return */ @OnMessage public String onMessage(@PathParam("relationId") String relationId, @PathParam("userCode") int userCode, String message) { return "Got your message (" + message + ").Thanks !"; } /** * 異常時觸發 * @param relationId * @param userCode * @param session */ @OnError public void onError(@PathParam("relationId") String relationId, @PathParam("userCode") int userCode, Throwable throwable, Session session) { log.info("Websocket Connection Exception: " + SessionUtils.getKey(relationId, userCode)); log.info(throwable.getMessage(), throwable); SessionUtils.remove(relationId, userCode); } /** * 關閉連線時觸發 * @param relationId * @param userCode * @param session */ @OnClose public void onClose(@PathParam("relationId") String relationId, @PathParam("userCode") int userCode, Session session) { log.info("Websocket Close Connection: " + SessionUtils.getKey(relationId, userCode)); SessionUtils.remove(relationId, userCode); } }
工具類用來儲存唯一key和連線
這個是我業務的需要,我的業務是伺服器有對應動作觸發時,推送資料到客戶端,沒有接收客戶端資料的操作。
import javax.websocket.Session; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; /** * 功能說明:用來儲存業務定義的sessionId和連線的對應關係 * 利用業務邏輯中組裝的sessionId獲取有效連線後進行後續操作 * 作者:liuxing(2014-12-26 02:32) */ public class SessionUtils { public static Map<String, Session> clients = new ConcurrentHashMap<>(); public static void put(String relationId, int userCode, Session session){ clients.put(getKey(relationId, userCode), session); } public static Session get(String relationId, int userCode){ return clients.get(getKey(relationId, userCode)); } public static void remove(String relationId, int userCode){ clients.remove(getKey(relationId, userCode)); } /** * 判斷是否有連線 * @param relationId * @param userCode * @return */ public static boolean hasConnection(String relationId, int userCode) { return clients.containsKey(getKey(relationId, userCode)); } /** * 組裝唯一識別的key * @param relationId * @param userCode * @return */ public static String getKey(String relationId, int userCode) { return relationId + "_" + userCode; } }
推送資料到客戶端
在其他業務方法中呼叫
/** * 將資料傳回客戶端 * 非同步的方式 * @param relationId * @param userCode * @param message */ public void broadcast(String relationId, int userCode, String message) { if (TelSocketSessionUtils.hasConnection(relationId, userCode)) { TelSocketSessionUtils.get(relationId, userCode).getAsyncRemote().sendText(message); } else { throw new NullPointerException(TelSocketSessionUtils.getKey(relationId, userCode) + " Connection does not exist"); } }
我是使用非同步的方法推送資料,還有同步的方法
詳見:http://docs.oracle.com/javaee/7/api/javax/websocket/Session.html
客戶端程式碼
var webSocket = null; var tryTime = 0; $(function () { initSocket(); window.onbeforeunload = function () { //離開頁面時的其他操作 }; }); /** * 初始化websocket,建立連線 */ function initSocket() { if (!window.WebSocket) { alert("您的瀏覽器不支援websocket!"); return false; } webSocket = new WebSocket("ws://127.0.0.1:8080/websocket.ws/" + relationId + "/" + userCode); // 收到服務端訊息 webSocket.onmessage = function (msg) { console.log(msg); }; // 異常 webSocket.onerror = function (event) { console.log(event); }; // 建立連線 webSocket.onopen = function (event) { console.log(event); }; // 斷線重連 webSocket.onclose = function () { // 重試10次,每次之間間隔10秒 if (tryTime < 10) { setTimeout(function () { webSocket = null; tryTime++; initSocket(); }, 500); } else { tryTime = 0; } }; }
其他除錯工具
Java實現一個websocket的客戶端
依賴:
<dependency> <groupId>org.java-websocket</groupId> <artifactId>Java-WebSocket</artifactId> <version>1.3.0</version> </dependency>
程式碼:
import java.io.IOException; import javax.websocket.ClientEndpoint; import javax.websocket.OnError; import javax.websocket.OnMessage; import javax.websocket.OnOpen; import javax.websocket.Session; @ClientEndpoint public class MyClient { @OnOpen public void onOpen(Session session) { System.out.println("Connected to endpoint: " + session.getBasicRemote()); try { session.getBasicRemote().sendText("Hello"); } catch (IOException ex) { } } @OnMessage public void onMessage(String message) { System.out.println(message); } @OnError public void onError(Throwable t) { t.printStackTrace(); } }
import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.net.URI; import javax.websocket.ContainerProvider; import javax.websocket.DeploymentException; import javax.websocket.Session; import javax.websocket.WebSocketContainer; public class MyClientApp { public Session session; protected void start() { WebSocketContainer container = ContainerProvider.getWebSocketContainer(); String uri = "ws://127.0.0.1:8080/websocket.ws/relationId/12345"; System.out.println("Connecting to " + uri); try { session = container.connectToServer(MyClient.class, URI.create(uri)); } catch (DeploymentException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } public static void main(String args[]){ MyClientApp client = new MyClientApp(); client.start(); BufferedReader br = new BufferedReader(new InputStreamReader(System.in)); String input = ""; try { do{ input = br.readLine(); if(!input.equals("exit")) client.session.getBasicRemote().sendText(input); }while(!input.equals("exit")); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
chrome安裝一個websocket客戶端除錯
最後
為了統一的操作體驗,對於一些不支援websocket的瀏覽器,請使用socketjs技術做客戶端開發。
相關文章
- 【轉】java中註解的使用與例項Java
- websocket簡單例項Web單例
- JAVA中動態性例項解釋 (轉)Java
- WebSocket 簡介及應用例項Web
- Mqtt websocket javascript 客戶端例項MQQTWebJavaScript客戶端
- tcl/tk例項詳解——glob使用例解
- java中String類常用方法的使用與例項Java
- Vue 原始碼解讀(6)—— 例項方法Vue原始碼
- spring boot +WebSocket 廣播式例項Spring BootWeb
- Java開發中的事件驅動模型例項詳解Java事件模型
- 例項講解Java工廠模式Java模式
- oracle 例項啟動和關閉解讀Oracle
- linux中sleep詳解例項Linux
- Java中內部類的例項化Java
- Linux中printf命令使用例項Linux
- 辛星解讀一次在mysql中獲取排名的例項MySql
- Java例項教程Java
- java介面例項Java
- html5+go+websocket簡單例項程式碼HTMLGoWeb單例
- Vue.js 2.0中$on與$emit如何使用之例項講解Vue.jsMIT
- 例項解說Linux中fdisk分割槽使用方法(轉)Linux
- java中父類宣告子類例項化Java
- Java中建立泛型型別的例項Java泛型型別
- Linux 中的 JQ 命令使用例項Linux
- Linux中ip命令的使用例項Linux
- vue例項中watch屬性的使用Vue
- Linux 中 ss 命令的使用例項Linux
- Linux中的basename命令使用例項Linux
- MySQL共享鎖:使用與例項詳解MySql
- Java使用jxl.jar匯出Excel例項JavaJARExcel
- C# WebSocket Fleck 原始碼解讀C#Web原始碼
- 基於 Node、WebSocket 的手機控制電腦例項Web
- java多型例項Java多型
- Thrift-java例項Java
- Java--例項化Java
- 淺談Java中的例項初始化器Java
- pinctrl使用例項
- 例項詳解 Java 死鎖與破解死鎖Java