Sring MVC 模式下使用websocket

前言

九風(fēng)的上一篇文章java WebSocket開發(fā)入門WebSocket介紹了websocket一些特征和提供了一個(gè)簡單的demo,但上篇文章中的websocket類不能被Spring MVC中的controller層、server層調(diào)用,這篇文章提供了一個(gè)可用于MVC模式調(diào)用的版本。

程序運(yùn)行環(huán)境說明

Server version : Apache Tomcat/7.0.65
JVM version: 1.7.0_75-b13

官網(wǎng)介紹

首先推薦大家看官網(wǎng)Oracle的javax.websocket介紹 ,官網(wǎng)對(duì)其中的ServerEndpoint的介紹是:ServerEndpoint是web socket的一個(gè)端點(diǎn),用于部署在URI空間的web socket服務(wù),該注釋允許開發(fā)者去定義公共的URL或URI模板端點(diǎn)和其他一些websocekt運(yùn)行時(shí)的重要屬性,例如encoder屬性用于發(fā)送消息。

上篇文章中的最簡單的tomcat的websocket服務(wù)器如果需要在MVC模式中被其他類調(diào)用,需要配置Configurator屬性:用于調(diào)用一些自定義的配置算法,如攔截連接握手或可用于被每個(gè)端點(diǎn)實(shí)例調(diào)用的任意的方法和算法;對(duì)于服務(wù)加載程序,該接口必須提供默認(rèn)配置器加載平臺(tái)。

spring MVC模式配置

對(duì)于spring MVC模式,ServerEndpoint的代碼為:

@ServerEndpoint(value="/websocketTest/{userId}",configurator = SpringConfigurator.class)

需要在maven中導(dǎo)入SpringConfiguator的包:

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-websocket</artifactId>
    <version>${spring.version}</version>
</dependency>

目標(biāo)需求

平臺(tái)有多個(gè)用戶,每個(gè)用戶可能會(huì)在多個(gè)終端上登錄,當(dāng)有消息到達(dá)時(shí),實(shí)時(shí)將所有消息推送給所有所用登錄的當(dāng)前用戶,就像QQ消息一樣,用戶在PC、pad、phone上同時(shí)登錄時(shí),將收到的消息同時(shí)推送給該用戶的所有終端。

數(shù)據(jù)結(jié)構(gòu)設(shè)計(jì)

每個(gè)用戶對(duì)應(yīng)多個(gè)終端,多個(gè)終端用Set來記錄,用戶使用Map來記錄,所以數(shù)據(jù)存儲(chǔ)結(jié)構(gòu)為Map<userId, Set<Terminal>>,結(jié)合具體的數(shù)據(jù)類型就是如下所示的的存儲(chǔ)結(jié)構(gòu):

//記錄每個(gè)用戶下多個(gè)終端的連接
private static Map<Long, Set<WebsocketDemo>> userSocket = new HashMap<>();

操作設(shè)計(jì)

  1. onOpen: 新建連接時(shí)需要判斷是否是該用戶第一次連接,如果是第一次連接,則對(duì)Map增加一個(gè)userId;否則將sessionid添加入已有的用戶Set中。
    2.onClose:連接關(guān)閉時(shí)將該用戶的Set中記錄remove,如果該用戶沒有終端登錄了,則移除Map中該用戶的記錄
  2. onMessage: 這個(gè)根據(jù)業(yè)務(wù)情況詳細(xì)設(shè)計(jì)。
  3. 給用戶的所有終端發(fā)送數(shù)據(jù):遍歷該用戶的Set中的連接即可。

后臺(tái)websocket連接代碼

/**
 * @Class: WebsocketDemo
 * @Description:  給所用戶所有終端推送消息
 * @author JFPZ
 * @date 2017年5月15日 上午21:38:08
 */
import java.io.IOException;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import javax.websocket.OnClose;
import javax.websocket.OnError;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.socket.server.standard.SpringConfigurator;

//websocket連接URL地址和可被調(diào)用配置
@ServerEndpoint(value="/websocketDemo/{userId}",configurator = SpringConfigurator.class)
public class WebsocketDemo {
    //日志記錄      
    private Logger logger = LoggerFactory.getLogger(WebsocketDemo.class);
    //靜態(tài)變量,用來記錄當(dāng)前在線連接數(shù)。應(yīng)該把它設(shè)計(jì)成線程安全的。
    private static int onlineCount = 0;
   
    //記錄每個(gè)用戶下多個(gè)終端的連接
    private static Map<Long, Set<WebsocketDemo>> userSocket = new HashMap<>();
 
    //需要session來對(duì)用戶發(fā)送數(shù)據(jù), 獲取連接特征userId
    private Session session;
    private Long userId;
   
    /**
     * @Title: onOpen
     * @Description: websocekt連接建立時(shí)的操作
     * @param @param userId 用戶id
     * @param @param session websocket連接的session屬性
     * @param @throws IOException
     */
    @OnOpen
    public void onOpen(@PathParam("userId") Long userId,Session session) throws IOException{
        this.session = session;
        this.userId = userId;
        onlineCount++;
        //根據(jù)該用戶當(dāng)前是否已經(jīng)在別的終端登錄進(jìn)行添加操作
        if (userSocket.containsKey(this.userId)) {
            logger.debug("當(dāng)前用戶id:{}已有其他終端登錄",this.userId);
            userSocket.get(this.userId).add(this); //增加該用戶set中的連接實(shí)例
        }else {
            logger.debug("當(dāng)前用戶id:{}第一個(gè)終端登錄",this.userId);
            Set<WebsocketDemo> addUserSet = new HashSet<>();
            addUserSet.add(this);
            userSocket.put(this.userId, addUserSet);
        }
        logger.debug("用戶{}登錄的終端個(gè)數(shù)是為{}",userId,userSocket.get(this.userId).size());
        logger.debug("當(dāng)前在線用戶數(shù)為:{},所有終端個(gè)數(shù)為:{}",userSocket.size(),onlineCount);
    }
   
    /**
     * @Title: onClose
     * @Description: 連接關(guān)閉的操作
     */
    @OnClose
    public void onClose(){
        //移除當(dāng)前用戶終端登錄的websocket信息,如果該用戶的所有終端都下線了,則刪除該用戶的記錄
        if (userSocket.get(this.userId).size() == 0) {
            userSocket.remove(this.userId);
        }else{
            userSocket.get(this.userId).remove(this);
        }
        logger.debug("用戶{}登錄的終端個(gè)數(shù)是為{}",this.userId,userSocket.get(this.userId).size());
        logger.debug("當(dāng)前在線用戶數(shù)為:{},所有終端個(gè)數(shù)為:{}",userSocket.size(),onlineCount);
    }
   
    /**
     * @Title: onMessage
     * @Description: 收到消息后的操作
     * @param @param message 收到的消息
     * @param @param session 該連接的session屬性
     */
    @OnMessage
    public void onMessage(String message, Session session) {    
        logger.debug("收到來自用戶id為:{}的消息:{}",this.userId,message);
        if(session ==null)  logger.debug("session null");
    }
   
    /**
     * @Title: onError
     * @Description: 連接發(fā)生錯(cuò)誤時(shí)候的操作
     * @param @param session 該連接的session
     * @param @param error 發(fā)生的錯(cuò)誤
     */
    @OnError
    public void onError(Session session, Throwable error){
        logger.debug("用戶id為:{}的連接發(fā)送錯(cuò)誤",this.userId);
        error.printStackTrace();
    }
   
  /**
   * @Title: sendMessageToUser
   * @Description: 發(fā)送消息給用戶下的所有終端
   * @param @param userId 用戶id
   * @param @param message 發(fā)送的消息
   * @param @return 發(fā)送成功返回true,反則返回false
   */
    public Boolean sendMessageToUser(Long userId,String message){
        if (userSocket.containsKey(userId)) {
            logger.debug(" 給用戶id為:{}的所有終端發(fā)送消息:{}",userId,message);
            for (WebsocketDemo WS : userSocket.get(userId)) {
                logger.debug("sessionId為:{}",WS.session.getId());
                try {
                    WS.session.getBasicRemote().sendText(message);
                } catch (IOException e) {
                    e.printStackTrace();
                    logger.debug(" 給用戶id為:{}發(fā)送消息失敗",userId);
                    return false;
                }
            }
            return true;
        }
        logger.debug("發(fā)送錯(cuò)誤:當(dāng)前連接不包含id為:{}的用戶",userId);
        return false;
    }
  
}

Service層編寫

service層比較簡單,不多說,直接看代碼就行:

/** 
 * @Class: WebSocketMessageService
 * @Description:  使用webscoket連接向用戶發(fā)送信息
 * @author JFPZ
 * @date 2017年5月15日 上午20:17:01
 */
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;
import org.springside.modules.mapper.JsonMapper;
import com.hhxk.vrshop.entity.Message;
import com.hhxk.vrshop.web.webSocket.WebsocketDemo;
import com.hhxk.vrshop.web.webSocket.WebSocket;

@Service("webSocketMessageService")
public class WSMessageService {
    private Logger logger = LoggerFactory.getLogger(WSMessageService.class);
    //聲明websocket連接類
    private WebsocketDemo websocketDemo = new WebsocketDemo();

    /**
     * @Title: sendToAllTerminal
     * @Description: 調(diào)用websocket類給用戶下的所有終端發(fā)送消息
     * @param @param userId 用戶id
     * @param @param message 消息
     * @param @return 發(fā)送成功返回true,否則返回false
     */
    public Boolean sendToAllTerminal(Long userId,String message){   
        logger.debug("向用戶{}的消息:{}",userId,message);
        if(websocketDemo.sendMessageToUser(userId,message)){
            return true;
        }else{
            return false;
        }   
    }           
}

Controller層

Controller層也簡單,只需提供一個(gè)http請(qǐng)求的入口然后調(diào)用service即可,Controller層請(qǐng)求地址為:http://127.0.0.1:8080/your-project-name/message/TestWS

@Controller
@RequestMapping("/message")
public class MessageController extends BaseController{
    //websocket服務(wù)層調(diào)用類
    @Autowired
    private WSMessageService wsMessageService;

  //請(qǐng)求入口
    @RequestMapping(value="/TestWS",method=RequestMethod.GET)
    @ResponseBody
    public String TestWS(@RequestParam(value="userId",required=true) Long userId,
        @RequestParam(value="message",required=true) String message){
        logger.debug("收到發(fā)送請(qǐng)求,向用戶{}的消息:{}",userId,message);
        if(wsMessageService.sendToAllTerminal(userId, message)){
            return "發(fā)送成功";
        }else{
            return "發(fā)送失敗";
        }   
    }
}

前端連接代碼

html支持websocket,將下面代碼中的websocket地址中替換即可直接使用:

<!DOCTYPE html>
<html>

    <head lang="en">
        <meta charset="UTF-8">
        <script src="http://cdn.sockjs.org/sockjs-0.3.min.js"></script>
        <link rel="stylesheet" >
        <link rel="stylesheet" >
        <script src="http://cdn.bootcss.com/jquery/1.11.3/jquery.min.js"></script>
        <script src="http://cdn.bootcss.com/bootstrap/3.3.5/js/bootstrap.min.js"></script>
        <title>webSocket-用戶66</title>
        <script type="text/javascript">
            $(function() {
                var websocket;
                if('WebSocket' in window) {
                                        console.log("此瀏覽器支持websocket");
                    websocket = new WebSocket("ws://127.0.0.1:8080/your-project-name/websocketDemo/66");
                } else if('MozWebSocket' in window) {
                    alert("此瀏覽器只支持MozWebSocket");
                } else {
                    alert("此瀏覽器只支持SockJS");
                }
                websocket.onopen = function(evnt) {
                    $("#tou").html("鏈接服務(wù)器成功!")
                };
                websocket.onmessage = function(evnt) {
                    $("#msg").html($("#msg").html() + "<br/>" + evnt.data);
                };
                websocket.onerror = function(evnt) {};
                websocket.onclose = function(evnt) {
                    $("#tou").html("與服務(wù)器斷開了鏈接!")
                }
                $('#send').bind('click', function() {
                    send();
                });

                function send() {
                    if(websocket != null) {
                        var message = document.getElementById('message').value;
                        websocket.send(message);
                    } else {
                        alert('未與服務(wù)器鏈接.');
                    }
                }
            });
        </script>
    </head>

    <body>
        <div class="page-header" id="tou">
            webSocket多終端聊天測試
        </div>
        <div class="well" id="msg"></div>
        <div class="col-lg">
            <div class="input-group">
                <input type="text" class="form-control" placeholder="發(fā)送信息..." id="message">
                <span class="input-group-btn">
                    <button class="btn btn-default" type="button" id="send" >發(fā)送</button>
                </span>
            </div>
        </div>
    </body>

</html>

測試驗(yàn)證

使用用戶id為66登錄2個(gè)終端,用戶id為88登錄1個(gè)終端,服務(wù)器中可以看到登錄的信息,如下圖所示:

用戶連接后臺(tái)websocket服務(wù)器

首先驗(yàn)證用戶66的兩個(gè)終端接收是否能同時(shí)接收消息:


驗(yàn)證用戶66的多終端接收消息

驗(yàn)證用戶88接收消息:


驗(yàn)證用戶88接收消息

總結(jié)

websocekt接口在mvc模式中被service層調(diào)用需要在@ServerEndpoint中添加configurator屬性,而實(shí)際情況需要根據(jù)業(yè)務(wù)需求來進(jìn)行設(shè)計(jì)。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,565評(píng)論 6 539
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,115評(píng)論 3 423
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,577評(píng)論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,514評(píng)論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,234評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,621評(píng)論 1 326
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,641評(píng)論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,822評(píng)論 0 289
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,380評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,128評(píng)論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,319評(píng)論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,879評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,548評(píng)論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,970評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,229評(píng)論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,048評(píng)論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,285評(píng)論 2 376

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,693評(píng)論 25 708
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,909評(píng)論 6 342
  • 閣樓上的柔情 一聲輕喚 貓?zhí)ь^仰望, 微微愉悅踏石階 你遺忘,這是回憶 你不去想 如線的心事 眼中的咸,那是 死去...
    詩盡閱讀 247評(píng)論 0 0
  • 文/風(fēng)的衣裳 01 雨夜,獨(dú)自行走在這條已經(jīng)走了十年的街道上。這樣的天實(shí)在應(yīng)該開著車出來,可是我猶豫了片刻,卻還是...
    風(fēng)的衣裳閱讀 1,260評(píng)論 22 24
  • 朋友成甲今天給我抱怨說,明明自己本著樂于助人的心態(tài),去幫助同事的,可是最后被人說成了狗拿耗子。好心當(dāng)了驢肝肺,以后...
    白夜2017閱讀 419評(píng)論 0 0