提示:记录工作中遇到的需求及解决办法

文章目录

  • 前言
  • 一、代码
  • 二、功能说明
  • 三、使用场景


前言

前端通过WebSocket的单例模式实现实时通信效果


提示:以下是本篇文章正文内容,下面案例可供参考

一、代码

export default class SocketService {/*** 单例实例*/static instance = null;static get Instance() {if (!this.instance) {this.instance = new SocketService();}return this.instance;}// 连接状态标记isConnected = false;// 重试计数器reconnectCount = 0;// 最大重连次数maxReconnect = 5;// 重连延迟 (毫秒)reconnectDelay = 3000;// WebSocket 实例ws = null;// 消息回调函数映射表callbacks = new Map();/*** 初始化 WebSocket 连接* @param {string} url - WebSocket 服务器地址*/connect(url) {if (this.ws) {this.disconnect();}this.ws = new WebSocket(url);// 连接成功this.ws.onopen = () => {this.isConnected = true;this.reconnectCount = 0;console.log("WebSocket connected");};// 接收消息this.ws.onmessage = (msg) => {try {const parsed = JSON.parse(msg.data);const { event, data } = parsed;// 触发对应事件回调if (this.callbacks.has(event)) {this.callbacks.get(event).forEach(cb => cb(data));}} catch (e) {console.error("Message parsing failed", e);}};// 连接关闭this.ws.onclose = () => {this.isConnected = false;console.log("WebSocket disconnected");// 自动重连逻辑if (this.reconnectCount < this.maxReconnect) {setTimeout(() => {console.log(`Reconnecting... (${this.reconnectCount + 1}/${this.maxReconnect})`);this.reconnectCount++;this.connect(url);}, this.reconnectDelay);}};// 错误处理this.ws.onerror = (err) => {console.error("WebSocket error:", err);this.ws.close();};}/*** 断开 WebSocket 连接*/disconnect() {if (this.ws) {this.ws.close();this.ws = null;this.isConnected = false;}}/*** 注册事件监听器* @param {string} event - 事件名称* @param {Function} callback - 回调函数*/on(event, callback) {if (!this.callbacks.has(event)) {this.callbacks.set(event, []);}this.callbacks.get(event).push(callback);}/*** 发送消息* @param {string} event - 事件类型* @param {Object} data - 发送数据*/send(event, data) {if (this.isConnected) {const payload = JSON.stringify({ event, data });this.ws.send(payload);} else {console.error("Cannot send message - WebSocket not connected");}}/*** 移除事件监听器* @param {string} event - 事件名称* @param {Function} callback - 要移除的回调函数*/off(event, callback) {if (this.callbacks.has(event)) {const callbacks = this.callbacks.get(event).filter(cb => cb !== callback);this.callbacks.set(event, callbacks);}}
}// 使用示例:
// const socket = SocketService.Instance;
// socket.connect("wss://api.example.com");
// socket.on("message", (data) => console.log("Received:", data));
// socket.send("update", { value: 42 });

二、功能说明

  1. 单例模式:
    • 通过静态属性 instance 和静态方法 get Instance() 确保全局只有一个实例
    • 首次调用 SocketService.Instance 时自动创建实例
  2. 核心功能:
    • connect(url):建立 WebSocket 连接
    • disconnect():主动断开连接
    • on(event, callback):注册事件监听器
    • off(event, callback):移除事件监听器
    • send(event, data):发送结构化数据
  3. 自动重连机制:
    • 连接断开时自动尝试重连(最多 5 次)
    • 每次重连间隔 3 秒
    • 连接成功后重置重连计数器
  4. 消息协议:
    • 使用 JSON 格式:{ event: ‘事件名’, data: … }
    • 自动解析消息并触发对应事件回调

三、使用场景

// 在应用初始化时建立连接
const socket = SocketService.Instance;
socket.connect("wss://api.example.com/ws");// 监听服务器消息
socket.on("chatMessage", (msg) => {console.log("New message:", msg);
});// 发送数据
document.getElementById("sendBtn").addEventListener("click", () => {socket.send("userAction", { type: "click", count: 5 });
});// 组件卸载时移除监听
const handler = (data) => { /* ... */ };
socket.on("update", handler);// 需要时移除特定监听器
socket.off("update", handler);

此实现提供了完整的 WebSocket 连接管理,包括自动重连、事件订阅/发布机制,并严格遵循单例模式确保全局状态一致性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/911883.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/911883.shtml
英文地址,请注明出处:http://en.pswp.cn/news/911883.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【缓存技术】深入分析如果使用好缓存及注意事项

Java 架构师缓存深度实践指南&#xff1a;策略、陷阱与高并发场景实战 一、缓存设计核心策略 1. 缓存选型与场景适配 缓存选型需结合业务场景、数据规模、性能要求等多维度评估&#xff1a; 场景推荐方案工具/技术案例高频读、极少写本地缓存Caffeine、Guava Cache电商平台商…

wordpress Contact Form 7表单插件设置使用教程

在wordpress安装插件Contact Form 7后&#xff0c;前端的提示信息会根据后台的语言决定。如果你安装的是版本的wordpress程序&#xff0c;出来的提示信息就是英文的。如果你安装的是中文版的wordpress程序&#xff0c;出来的提示信息就是中文的。前端显示什么样的语言取决你安装…

Qt实现tcp通信(QTcpServer和QTcpSocket的应用)详细教程

Qt实现tcp通信&#xff08;QTcpServer和QTcpSocket的应用&#xff09;详细教程 服务端 监听地址和端口 ip可以是Ipv4Any&#xff0c;本机地址&#xff0c;也可以是固定的某个ip 端口号则作为服务端绑定的端口&#xff0c;客户端连接服务端时需要连接到服务端绑定的端口&#x…

Spring Boot Actuator 跟踪HTTP请求和响应

一、Spring Boot Actuator简介 什么是Spring Boot Actuator&#xff1f;Spring Boot Actuator 是 Spring Boot 提供的生产级监控和管理模块&#xff0c;旨在帮助开发者实时监控应用状态、收集运行时指标&#xff0c;并提供一系列管理端点&#xff08;Endpoints&#xff09;&am…

Java底层原理:深入理解线程与并发机制

一、线程的基本概念 线程是操作系统能够进行运算调度的最小单位。它被包含在进程之中&#xff0c;是进程中的实际运作单位。一个进程可以包含多个线程&#xff0c;这些线程共享进程的资源&#xff0c;但每个线程有其独立的执行路径。 &#xff08;一&#xff09;线程与进程的…

使用Dockerfile定制化镜像和私有网盘案例

使用Dockerfile定制化镜像和私有网盘案例 镜像的分层结构是docker很重要的环节&#xff0c;如同搭积木一样。 也主要是为了节省资源。 理解docker build的工作原理 Docker在运行的时候分为Docker引擎&#xff0c;和客户端工具。docker的引擎提供一组API&#xff0c;被称为Docke…

分享一些实用的PHP函数(对比js/ts实现)

检查数组所有元素是否满足给定条件 如果提供的函数对数组的所有元素返回 true&#xff0c;则返回 true&#xff0c;否则返回 false。 思路 实现思路如下: 使用数组的filter方法对数组执行给定的函数&#xff0c;然后使用count方法获取执行后的结果&#xff0c;再和count方法…

HuggingFace下载的模型缓存到了C盘,如何安全迁移到其他盘

HuggingFace下载的模型缓存到了C盘&#xff0c;如何安全迁移到其他盘 ✅ 方法一&#xff1a;通过环境变量永久迁移缓存目录&#xff08;**推荐**&#xff09;步骤如下&#xff1a;1️⃣ 在 D 盘创建一个目录&#xff08;比如&#xff09;&#xff1a;2️⃣ 设置环境变量&#x…

vue 开启 source-map 后构建速度会很慢

vue.config.js 的 configureWebpack.devtool 设置为 source-map 后&#xff0c;编译速度会变慢&#xff0c;严重影响开发进度。 改善方法&#xff1a; 一&#xff1a;改为 module-eval-source-map 改为 module-eval-source-map 在打包速度和源码信息还原程度都属于良好程度。…

三维空间数据格式全景图:OSGB | 3Dtiles | I3S | S3M

前言 当城市数字孪生遇上国产化替代&#xff0c;开发者该如何选型&#xff1f; ——深入解析四大主流格式的技术基因与落地场景 &#x1f52e; 趋势预言&#xff1a; 2025年国土空间规划全面实景三维化 ▶ S3M市占率将突破60%Cesium开源生态持续领跑Web端 ▶ 3Dtiles仍是跨国项…

【服务器】服务器选型设计

一、服务器产品说明 1.1 基础服务器说明 1.1.1、服务器产品分类与核心指标​​ ​​1. 通用分类体系​​ ​​类型​​ ​​适用场景​​ ​​代表产品指标​​ 塔式服务器 中小企业应用 ≤2路CPU&#xff0c;32G内存&#xff0c;支持热插拔HDD 机架服务器 数据中心主…

【C++】noexcept的作用

【C】noexcept的作用 noexcept是C11引入的关键字&#xff0c;用于指定函数是否会抛出异常。它既是一个修饰符也是一个操作符&#xff0c;在现代C编程中扮演着重要角色。 一、noexcept的基本概念 noexcept主要有两种形式&#xff1a; 无条件形式​&#xff1a;void func() n…

Docker 到底是个什么?

Docker 技术全面解析&#xff1a;从安装到应用 一、引言 在当今的软件开发和运维领域&#xff0c;Docker 已经成为了一个不可或缺的工具。它以其轻量级、可移植性和高效性等特点&#xff0c;改变了开发者和运维人员的工作方式&#xff0c;使得应用的打包、分发、运行变得更加…

Python 可迭代的对象、迭代器 和生成器(Sentence类第3版:生成器函数)

Sentence类第3版&#xff1a;生成器函数 实现相同功能&#xff0c;但却符合 Python 习惯的方式是&#xff0c;用生成器函数代替 SentenceIterator 类。先看示例 14-5&#xff0c;然后详细说明生成器函数。 示例 14-5 sentence_gen.py&#xff1a;使用生成器函数实现 Sentence…

5G自协商

好的&#xff0c;下面是一个基于裸机 C 环境的自协商实现示例代码&#xff0c;支持 **最高 5G 并向下兼容**。这个代码框架假设你使用的是 IEEE 802.3 规范下的 **MDIO** 接口和常见的 **Marvell PHY**&#xff08;或类似支持 5G/2.5G 的PHY&#xff09;。 c #include <std…

sublime 4200 激活

目录 下载激活方式一&#xff1a;sublime 打开方式二&#xff1a;https://hexed.it 打开 下载 sublime官方下载 sublime_text_build_4200_x64.zip 激活 方式一&#xff1a;sublime 打开 1、复制 sublime_text.exe 为 sublime_text_activation.exe 2、直接使用 sublime_ 打开…

Ceph 和 MinIO

Ceph 和 MinIO 是两种主流的分布式存储系统&#xff0c;但它们的设计目标、架构和使用场景存在显著差异。以下是清晰的对比解析&#xff1a; &#x1f9e9; 核心定位对比 维度CephMinIO定位统一存储平台&#xff08;块/对象/文件&#xff09;纯对象存储&#xff08;S3兼容&…

cili3d笔记20 正交投影3d重建笔记1

正交视图转3d mostFrequentCluster.lines.forEach(line > {const [x1, y1, x2, y2] line;let xhat{x1,x2};let yhat{y1,y2};}); 没考虑到侧视图 const clusters clusterLines(inputlines, 5);const lines3d:[number,number,number,number,number,number][][]const { mostM…

【Docker基础】Docker容器生命周期管理:从创建到删除的完整指南

目录 1 容器生命周期概述 2 容器创建&#xff08;docker create&#xff09; 2.1 docker create命令详解 2.2 创建流程解析 2.3 创建与运行的区别 3 容器启动&#xff08;docker start&#xff09; 3.1 docker start命令详解 3.2 启动流程解析 3.3 启动与运行的区别 …

Docker、Docker composer与Docker desktop

文章目录 Docker、Docker composer与Docker desktop一、docker、docker composer、docker desktop1.1 Docker1.2 Docker Compose1.3 Docker Desktop1.4 三者之间的区别 二、docker desktop的安装和换源2.1 前期准备WSL2 2.1 Docker Desktop 安装下载 Docker Desktop安装 Docker…