WebWorker與WebSocket實現前端消息匯流排

来源:https://www.cnblogs.com/yfrs/archive/2018/08/10/work_ws.html
-Advertisement-
Play Games

Web Worker讓JS有了多線程的能力,可以將複雜耗時的操作都交付給Worker線程處理。WebSocket讓web端與服務端維持一個有效的長連接,實現服務端主動推送數據。將二者一結合,業務系統信息流轉通知功能完全就可以剝離出來。 架構圖 JS Worker Worker工作在一個專用的作用域D ...


Web Worker讓JS有了多線程的能力,可以將複雜耗時的操作都交付給Worker線程處理。WebSocket讓web端與服務端維持一個有效的長連接,實現服務端主動推送數據。將二者一結合,業務系統信息流轉通知功能完全就可以剝離出來。

架構圖

Alt text

JS Worker

Worker工作在一個專用的作用域DedicatedWorkerGlobalScope,在這個作用域中,不能直接操作DOM節點,不能使用Window對象的預設方法和屬性。不過對於網路的訪問是完全沒有問題的。具體能使用那些對象和方法請點擊這裡查看

從上圖中可明顯的看出,Worker在當前架構中實現一個橋梁的左右,上連接socket端中的數據,下負責分發socket中的數據。此處我們先瞭解下Worker本身的功能實現。

  1. 主線程與Worker線程通過方法postMessage相互傳遞信息
  2. 主線程與Worker線程通過事件onmessage接收相互傳遞的消息
  3. Worker中引入第三方js使用方法importScripts([url,])
  4. 主線程調用worker.terminate()結束線程
  5. Worker線程通過調用this.close()結束自身線程

新建一個webworker.js文件,併在其中編寫如下代碼

//author:herbert qq:464884492
onmessage = function (event) {
    if (event.data.code) {
        var code = event.data.code.toLowerCase();
        switch (code) {
            case "init":
                var userId = event.data.loggedUserId;
                var sessionId = event.data.sessionid;
                if (!sessionId) {
                    this.close();
                    return;
                }
                postMessage({ code: "codeone", msg: "你好,組件1" });
                postMessage({ code: "codetwo", msg: "你好,組件2" });
                break;
            default:
                break;
        }
    }
}

註意:在 onmessage 前不能加var否則在IE下會接收不了消息。IE總是讓你充滿挫敗感!!

新建一個index.html頁面,在script塊中編寫以下代碼,實現與webworker.js通訊

//author:herbert qq:464884492
var work = new Worker('webworker.js')
    , textone = document.querySelector("#textone")
    , textTwo = document.querySelector("#texttwo")
textAll = document.querySelector("#textAll");
work.onmessage = function (event) {
    var data = event.data;
    if (!!data.code) {
        switch (data.code) {
            case "close":
                work.terminate();
            case "codeone":
                textone.value = textone.value + JSON.stringify(data) + "\r\n";
                textAll.value = textAll.value + JSON.stringify(data) + "\r\n";
                break;
            case "codetwo":
                textTwo.value = textTwo.value + JSON.stringify(data) + "\r\n";
                textAll.value = textAll.value + JSON.stringify(data) + "\r\n";
                break;
            default:
                textAll.value = textAll.value + JSON.stringify(data) + "\r\n";
        }
    }
};
work.postMessage({
    code: "init",
    loggedUserId: 'demo',
    sessionid: 'demo'
});

JS WebSocket

WebSocket和Http一樣都是基於Tcp協議。不同是WebSocket實現了服務端與客戶端的全雙工通訊。在Websocket未出現之前,要是實現一個信息推送的功能,通過http來實現唯一方案就是輪訓,輪訓分長短,各有弊端。現在WebSocket一齣現,一切都好辦了。

接下來我們開始建立一個WebSocket連接

方法中的root表示當前作用域,在主線程是root=window,在WebWorker線程root=DedicatedWorkerGlobalScope

//author:herbert qq:464884492
var root = this, socket = null;
function connect(wsurl) {
    if ('WebSocket' in root) {
        socket = new WebSocket(wsurl);
    } else if ('MozWebSocket' in root) {
        socket = new MozWebSocket(wsurl);
    } else {
        alert("您的瀏覽器版本過低,將不能接收系統消息");
    }
}

wsurl格式為ws:\\ 或者 wss:\\,後者表示SSL加密傳輸。實際地址如: ws://localhost:8090/demo/demowebsocket
接下來,我們需要為socket處理事件,負責接收服務端推送的消息

//author:herbert qq:464884492
function onOpen() {
    postMessage({ code: "openConnect" });
}
function onClose() {
    postMessage({ code: "closewsconnect" });
}
function onMessaage(event) {
    postMessage(JSON.parse(event.data));
}
function onError(event) {
    socket = null;
    if (event.target.readyState == 3) {
        //斷線重連
        setTimeout(function () {
            connect(event.target.url);
            initMessageEvent();
        }, 1000);
    }
}
function sendMessage(msg) {
    if (socket == null) return;
    socket.send(msg);
}
function initMessageEvent() {
    socket.onopen = onOpen; //socket連接成功處理事件
    socket.onclose = onClose; //socket連接關閉處理事件
    socket.onmessage = onMessaage; //socket接收到新消息
    socket.onerror = onError; //soket錯誤處理事件
}

JAVA WebSocket

Tomcat7x已經實現了標準WebScoket介面,在項目中只需要編寫一個普通的實體bean配置註解就可以實現一個標準的WebSocket Api。開發中主要使用一些註解

  • @ServerEndpoint 設置WebSocket連接地址,以及url參數
    如: @ServerEndpoint(value = “/demowebsocket/{userId}/{sessionId}”),其中{userId}、{sessionId} 為pathParam可以在onOpen函數中通過函數參數 @PathParam 獲取
  • @PathParam 獲取URL地址上對應的註解參數
  • @OnOpen 建立連接註解
  • @OnClose 關閉連接註解
  • @OnMessage 接收消息註解
  • @OnError 錯誤註解

被註解約束的函數都可以任意選擇需要的參數,可選擇的參數有 Session、EndpointConfig 以及 @PathParam, 服務端Bean代碼如下

//author:herbert qq:464884492
@ServerEndpoint(value = "/demowebsocket/{userId}/{sessionId}")
public class DemoWebSokcet {
    private static final Set<DemoWebSokcet> connections = new CopyOnWriteArraySet<DemoWebSokcet>();
    private Session session;
    public DemoWebSokcet() {
    }

    @OnOpen
    public void openConnection(Session session, EndpointConfig conf,
        @PathParam("userId") String userId,
        @PathParam("sessionId") String sessionId) {
        this.session = session;
        connections.add(this);
        JSONObject jo = new JSONObject();
        jo.put("code", "newuser");
        jo.put("userid", userId);
        jo.put("sessionid", sessionId);
        jo.put("msg", "server:新連接用戶");
        sendMessage(jo);

        // 測試 代碼
        JSONObject jo1 = new JSONObject();
        jo1.put("code", "codeone");
        jo1.put("userid", userId);
        jo1.put("sessionid", sessionId);
        jo1.put("msg", "Server:組件1你好");
        sendMessage(jo1);

        JSONObject jo2 = new JSONObject();
        jo2.put("code", "codetwo");
        jo2.put("userid", userId);
        jo2.put("sessionid", sessionId);
        jo2.put("msg", "server:組件2你好");
        sendMessage(jo2);
    }

    @OnClose
    public void closeConnection(@PathParam("userId") String userId,
        @PathParam("sessionId") String sessionId) {
        connections.remove(this);
        JSONObject jo = new JSONObject();
        jo.put("code", "connectionClose");
        jo.put("userid", userId);
        jo.put("sessionid", sessionId);
        jo.put("msg", "server:連接關閉");
        sendMessage(jo);
    }

    // 處理文本消息
    @OnMessage
    public void handleTextMsg(Session session, String message,
        @PathParam("userId") String userId,
        @PathParam("sessionId") String sessionId) {
        System.out.println("userId=>" + userId + " sessionId=>" + sessionId);
        // 原樣轉發客戶端消息
        sendMessage(JSONObject.parseObject(message));
    }

    // 處理二進位消息
    @OnMessage
    public void handleBinaryMsg(Session session, ByteBuffer msg,
        @PathParam("userId") String userId,
        @PathParam("sessionId") String sessionId) {

    }

    // 處理pong消息
    @OnMessage
    public void handlePongMsg(Session session, PongMessage msg,
        @PathParam("userId") String userId,
        @PathParam("sessionId") String sessionId) {
        JSONObject jo = new JSONObject();
        jo.put("code", "pong");
        jo.put("userid", userId);
        jo.put("sessionid", sessionId);
        jo.put("msg", msg.getApplicationData().toString());
        sendMessage(jo);
    }

    @OnError
    public void onError(Throwable t, @PathParam("userId") String userId,
        @PathParam("sessionId") String sessionId) throws Throwable {
    JSONObject jo = new JSONObject();
    jo.put("code", "servererror");
    jo.put("userid", userId);
    jo.put("sessionid", userId);
    jo.put("msg", t.getMessage());
    sendMessage(jo);
}
 
private static void sendMessage(JSONObject msg) {
    for (DemoWebSokcet client : connections) {
        try {
            synchronized(client) {
                client.session.getBasicRemote()
                    .sendText(msg.toJSONString());
            }
        } catch (IOException e) {
            JSONObject jo = new JSONObject();
            jo.put("code", "servererror");
            jo.put("userid",
                client.session.getPathParameters().get("userid"));
            jo.put("sessionid",
                client.session.getPathParameters().get("sessionid"));
            connections.remove(client);
            try {
                client.session.close();
            } catch (IOException e1) {
            }

            jo.put("msg", "server:發送消息出現異常,連接已關閉" + e.getMessage());
            sendMessage(jo);
        }
    }
}
}

在測試代碼編寫過程中,通過pom方式引入javax.websocket-api,啟動後始終出現 Error during WebSocket handshake: Unexpected response code: 404連接錯誤,後來通過直接件tomcat/bin下對應的tomcat實現的jar複製到webapp對應的bin文件夾下解決問題。

Demo預覽

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 開題先拋一個快應用的開發文檔鏈接 https://doc.quickapp.cn/ 我只能說這個文檔不是很人性化,左側導航欄分了『指南和參考』,結果我最關心的組件,在『指南』中只有list和tabs,其餘組件都在『參考』中,真是噴了一口老血才找到。 根據開發微信小程式和支付寶小程式的經驗,我習慣性的 ...
  • JavaScript 對象是動態的屬性“包”(指其自己的屬性)。JavaScript 對象有一個指向一個原型對象的鏈。當試圖訪問一個對象的屬性時,它不僅僅在該對象上搜尋,還會搜尋該對象的原型,以及該對象的原型的原型,依次層層向上搜索,直到找到一個名字匹配的屬性或到達原型鏈的末尾。 遵循ECMAScr ...
  • 彈窗的工作原理:在網頁中寫一個div ,佈局到想要顯示的位置,將display設為none,隱藏該div。然後通過點擊事件或其他操作,利用Js代碼,將display設置為block,將div 顯示到網頁中。 Tips:display:none//隱藏 display: block//顯示 效果圖:點 ...
  • 此文是我的出版書籍《React Native 精解與實戰》連載分享,此書由機械工業出版社出版,書中詳解了 React Native 框架底層原理、React Native 組件佈局、組件與 API 的介紹與代碼實戰,以及 React Native 與 iOS、Android 平臺的混合開發底層原理講... ...
  • 1、 表單驗證:減輕伺服器的壓力、保證輸入的數據符合要求; 2、 常用的表單驗證:日期格式、表單元素是否為空、用戶名和密碼、E-mail地址、身份證號碼等; 3、 表單驗證的思路: 1. 獲得表單元素值,這些值一般是String類型,包含數字、下劃線等; 2. 使用JavaScript的一些方法對獲 ...
  • HTML原生就是響應式的(HTML內容在視口內流式的分佈)。隨著CSS的廣泛應用,設計者創建了許多固定佈局的“盒子”並把內容強制的放在盒子之中,這有悖於HTML原生響應的特性。 ...
  • //顏色16進位轉RGB方法 String.prototype.colorRgb = function(){ var sColor = this.toLowerCase(); //十六進位顏色值的正則表達式 var reg = /^#([0-9a-fA-f]{3}|[0-9a-fA-f]{6})$/... ...
  • var shine=0.8; var arrays = ['[255,182,193,0.8]','[144,238,144,0.8]','[255,235,205,0.8]','[240,128,128,0.8]','[255,186,0,0.8]','[255,225,189,0.8]','[2... ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...