单例模式加闭包实现 WebSocket 全局唯一实例,核心是延迟初始化、闭包封装状态与防重复创建,通过模块顶层执行并导出统一实例,确保连接生命周期可控、状态可预期且避免资源浪费。
用单例模式加闭包实现 WebSocket 全局唯一实例,核心是“延迟初始化 + 闭包封装状态 + 防重复创建”。关键不在语法炫技,而在控制连接生命周期、避免多处 new WebSocket 导致资源浪费或状态混乱。
闭包能将 WebSocket 实例、重连逻辑、消息监听器等私有变量锁在函数作用域内,外部无法直接修改或覆盖。典型结构是返回一个对象,只暴露 connect、send、close 等受控方法:
const createWebSocket = () => { let instance = null; let reconnectTimer = null;<p>const connect = (url) => {if (instance && instance.readyState === WebSocket.OPEN) return;if (instance && (instance.readyState === WebSocket.CONNECTING || instance.readyState === WebSocket.OPEN)) {return; // 正在连接或已连上,不重复发起}</p><pre class="brush:php;toolbar:false;">instance = new WebSocket(url);instance.onopen = () => { /* 清除重连计时器 */ };instance.onmessage = (e) => { /* 统一处理消息 */ };instance.onerror = () => { /* 触发重连 */ };instance.onclose = () => { /* 延迟重连 */ };
};
return {connect,send: (data) => instance?.readyState === WebSocket.OPEN && instance.send(data),close: () => instance?.close(),getReadyState: () => instance?.readyState ?? 0};};
const ws = createWebSocket(); // 执行一次,返回带方法的对象
真正实现“单例”,靠的是模块级导出或全局命名空间绑定——不是靠类的 static 实例,而是靠执行时机和导出方式:
单例的意义不只是“只有一个”,更是“状态可预期”。需主动管理 readyState、错误恢复、发送队列(连接未就绪时暂存消息):
看似简单,实操中高频出错点集中在生命周期与引用泄漏: