日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

websocket后台推送数据

發(fā)布時(shí)間:2024/1/23 编程问答 22 豆豆
生活随笔 收集整理的這篇文章主要介紹了 websocket后台推送数据 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
介紹

現(xiàn)在很多網(wǎng)站為了實(shí)現(xiàn)即時(shí)通訊,所用的技術(shù)都是輪詢(polling)。輪詢是在特定的的時(shí)間間隔(如每1秒),

由瀏覽器對服務(wù)器發(fā)出HTTP request,然后由服務(wù)器返回最新的數(shù)據(jù)給客服端的瀏覽器。

這種傳統(tǒng)的HTTP request 的模式帶來很明顯的缺點(diǎn) – 瀏覽器需要不斷的向服務(wù)器發(fā)出請求,

然而HTTP request 的header是非常長的,里面包含的數(shù)據(jù)可能只是一個很小的值,這樣會占用很多的帶寬。

而最比較新的技術(shù)去做輪詢的效果是Comet – 用了AJAX。但這種技術(shù)雖然可達(dá)到全雙工通信,但依然需要發(fā)出請求。

在 WebSocket API,瀏覽器和服務(wù)器只需要要做一個握手的動作,然后,瀏覽器和服務(wù)器之間就形成了一條快速通道。兩者之間就直接可以數(shù)據(jù)互相傳送。

運(yùn)行環(huán)境: 客戶端

實(shí)現(xiàn)了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+ 服務(wù)端 依賴

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>

注意:早前業(yè)界沒有統(tǒng)一的標(biāo)準(zhǔn),各服務(wù)器都有各自的實(shí)現(xiàn),現(xiàn)在J2EE7的JSR356已經(jīng)定義了統(tǒng)一的標(biāo)準(zhǔn),請盡量使用支持最新通用標(biāo)準(zhǔn)的服務(wù)器。

詳見:

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自帶的實(shí)現(xiàn),后來要升級Tomcat 8,結(jié)果原來的實(shí)現(xiàn)方式在Tomcat 8不支持了,就只好切換到支持Websocket 1.0版本的Tomcat了。

主流的java?web服務(wù)器都有支持JSR365標(biāo)準(zhǔn)的版本了,請自行Google。

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

Tomcat的處理方式建議修改為NIO的方式,同時(shí)修改連接數(shù)到合適的參數(shù),請自行Google!

服務(wù)端不需要在web.xml中做額外的配置,Tomcat啟動后就可以直接連接了。

實(shí)現(xiàn) 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的標(biāo)準(zhǔn)
* 切忌直接在該連接處理類中加入業(yè)務(wù)處理代碼
* 作者:liuxing(2014-11-14 04:20)
*/
//relationId和userCode是我的業(yè)務(wù)標(biāo)識參數(shù),websocket.ws是連接的路徑,可以自行定義
@ServerEndpoint("/websocket.ws/{relationId}/{userCode}")
public class WebsocketEndPoint {
private static Log log = LogFactory.getLog(WebsocketEndPoint.class);
/**
* 打開連接時(shí)觸發(fā)
* @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);
}
/**
* 收到客戶端消息時(shí)觸發(fā)
* @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 !";
}
/**
* 異常時(shí)觸發(fā)
* @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);
}
/**
* 關(guān)閉連接時(shí)觸發(fā)
* @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和連接

這個是我業(yè)務(wù)的需要,我的業(yè)務(wù)是服務(wù)器有對應(yīng)動作觸發(fā)時(shí),推送數(shù)據(jù)到客戶端,沒有接收客戶端數(shù)據(jù)的操作。

import javax.websocket.Session;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
* 功能說明:用來存儲業(yè)務(wù)定義的sessionId和連接的對應(yīng)關(guān)系
* 利用業(yè)務(wù)邏輯中組裝的sessionId獲取有效連接后進(jìn)行后續(xù)操作
* 作者: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;
}
}

推送數(shù)據(jù)到客戶端

在其他業(yè)務(wù)方法中調(diào)用


/**
* 將數(shù)據(jù)傳回客戶端
* 異步的方式
* @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");
}
}

我是使用異步的方法推送數(shù)據(jù),還有同步的方法

詳見: http://docs.oracle.com/javaee/7/api/javax/websocket/Session.html

客戶端代碼

var webSocket = null;
var tryTime = 0;
$(function () {
initSocket();
window.onbeforeunload = function () {
//離開頁面時(shí)的其他操作
};
});
/**
* 初始化websocket,建立連接
*/
function initSocket() {
if (!window.WebSocket) {
alert("您的瀏覽器不支持websocket!");
return false;
}
webSocket = new WebSocket("ws://127.0.0.1:8080/websocket.ws/" + relationId + "/" + userCode);

// 收到服務(wù)端消息
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;
}
};
}

其他調(diào)試工具

Java實(shí)現(xiàn)一個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();?
}?
}?
} 創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅(jiān)持創(chuàng)作打卡瓜分現(xiàn)金大獎

總結(jié)

以上是生活随笔為你收集整理的websocket后台推送数据的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。