Web Worker让JS有了多线程的能力,可以将复杂耗时的操作都交付给Worker线程处理。WebSocket让web端与服务端维持一个有效的长连接,实现服务端主动推送数据。将二者一结合,业务系统信息流转通知功能完全就可以剥离出来。

架构图

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:464884492var 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:464884492function 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() {}@OnOpenpublic 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);}@OnClosepublic 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);}// 处理文本消息@OnMessagepublic 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));}// 处理二进制消息@OnMessagepublic void handleBinaryMsg(Session session, ByteBuffer msg,@PathParam("userId") String userId,@PathParam("sessionId") String sessionId) {}// 处理pong消息@OnMessagepublic 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);}@OnErrorpublic 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预览

总结

篇幅比较长,读到这里也不容易!WebWorker和WebSocket我也是第一次将二者结合起来。感觉现在javascript功能真的是越来越丰富了。demo地址,还有一点感悟,对于开发中的新知识点,首先你得学会怎么用,其次在通过阅读源码,以及理论知识让你使用的更顺利,甚至改变它。

WebWorker与WebSocket实现前端消息总线相关推荐

  1. 微前端架构实现(项目引入,消息总线,构建部署,监听服务)

    简介 为了解决庞大的一整块后端服务带来的变更与扩展方面的限制,出现了微服务架构(Microservices): 微服务是面向服务架构(SOA)的一种变体,把应用程序设计成一系列松耦合的细粒度服务,并通 ...

  2. netty服务器定时发送消息,netty+websocket+quartz实现消息定时推送

    netty+websocket+quartz实现消息定时推送&&IM聊天室 在讲功能实现之前,我们先来捋一下底层的原理,后面附上工程结构及代码 1.NIO NIO主要包含三大核心部分: ...

  3. springboot整合websocket实现一对一消息推送和广播消息推送

    springboot基础环境,请参考springboot文档 maven依赖 <dependency><groupId>org.springframework.boot< ...

  4. 基于netty搭建websocket,实现消息的主动推送

    基于netty搭建websocket,实现消息的主动推送 rpf_siwash https://www.jianshu.com/p/56216d1052d7 netty是由jboss提供的一款开源框架 ...

  5. vue项目中通过WebSocket实现实时消息提示及遇到的问题

    vue项目中通过WebSocket实现实时消息提示(前端代码) 需求说明 后台有新增消息通知,并实时推送给用户端,用websocket可以很方便的解决这个问题,但是websocket有个弊端不兼容IE ...

  6. SpringCloud之消息总线组件及微服务网关

    消息总线组件 Spring Cloud Bus 单个工程更新 有了配置中心,我们就可以吧配置文件放到git上来统一管理了,但是如果配置文件发生了变化,客户端 又如何更新呢? 1.在配置文件中增加自定义 ...

  7. StompJS+SpeechSynthesis实现前端消息实时语音播报

    前言 前端消息的实时推送我相信很多人不陌生,我们可以想到利用WebSocket,服务端主动向客户端推送数据,浏览器和服务器只需要完成一次握手,两者之间就直接可以创建持久性的连接,并进行双向数据传输.其 ...

  8. SpringCloud微服务-服务注册发现-负载均衡-服务调用-服务降级-服务网关-配置中心-消息总线-消息驱动-链路追踪-alibaba-nacos-sentinel-seata理论原理分析

    SpringCloud理论技术 概述 ​ Spring Cloud是一系列框架的有序集合.它利用Spring Boot的开发便利性巧妙地简化了分布式系统基础设施的开发,如服务发现注册.配置中心.消息总 ...

  9. 原 史上最简单的SpringCloud教程 | 第八篇: 消息总线(Spring Cloud Bus)(Finchley版本)

    转载请标明出处: 原文首发于:https://www.fangzhipeng.com/springcloud/2018/08/30/sc-f8-bus/ 本文出自方志朋的博客 转载请标明出处: Spr ...

最新文章

  1. iOS Socket Client 通讯
  2. 七基于Fourinone实现MQ demo
  3. 杭电多校(四)2019.7.31--暑假集训
  4. [C # 读书笔记]interface 接口 abstract
  5. linux取消中文网,SELinux如何关闭
  6. mongodb数据库常用指令
  7. 一次SocketException:Connection reset 异常排查
  8. 预训练 | 2022年 预训练的下一步是什么?
  9. DMSP/OLS夜间灯光数据
  10. 各种版本mysql驱动包下载地址
  11. 流程图伪代码计算机语言,论文中伪代码怎么写
  12. Excel图表制作(二):滚动条实现动态图表
  13. HTTP中GET,POST和PUT的区别
  14. shell获取脚本本身名称_linux,shell脚本中获取脚本的名字,使用脚本的名字。
  15. 「需求广场」需求词更新明细(九)
  16. 我为什么建议大家一定要考研?
  17. smss.exe是什么进程?详解Windows会话管理器中的smss.exe
  18. seaborn 频数统计直方图
  19. 通过RGB或YUV改变图像的色度和饱和度
  20. 计算机硬件耗电,耗电大户不一定就“费电”_主板评测-中关村在线

热门文章

  1. 直播回顾 | 最强中文NLP预训练模型艾尼ERNIE官方揭秘
  2. java-数组排序--冒泡排序、鸡尾酒排序、地精排序
  3. OpenStack部署
  4. 世界在音乐中得到了完整的再现和表达。
  5. 对进入单用户进行加密
  6. 揭秘百度核心技术:53位专家纯干货分享
  7. “TNS-03505:无法解析名称”问题解决一例
  8. Apache Spark源码走读之16 -- spark repl实现详解
  9. 关于ANDRID sdk安装过慢问题
  10. SQL2K数据库开发十一之表操作创建UNIQUE约束