websocket的封装

功能:
自动重连(断开自动重连)

心跳保活(防止服务器踢下线)

消息重发(发送失败重试)

错误统一处理

在线 / 离线监听

消息订阅 / 派发(解耦)

可重入、单例模式

javascript 复制代码
class WebSocketClient {
  constructor(url) {
    // 基础配置
    this.url = url
    this.socket = null
    this.isConnected = false

    // 重连配置
    this.reconnectEnabled = true // 自动重连开关
    this.reconnectDelay = 3000   // 重连间隔 3s
    this.reconnectTimer = null   // 重连定时器

    // 心跳配置(防止服务器断开)
    this.heartbeatEnabled = true
    this.heartbeatMsg = 'ping' // 心跳消息
    this.heartbeatInterval = 5000 // 5秒一次心跳
    this.heartbeatTimer = null

    // 消息订阅器(不同页面接收不同消息)
    this.listeners = new Map()

    // 缓存未发送成功的消息
    this.msgQueue = []
  }

  // ========== 1. 建立连接 ==========
  connect() {
    // 防止重复创建
    if (this.socket && this.socket.readyState === 1) return

    this.socket = new WebSocket(this.url)

    // 连接成功
    this.socket.onopen = (e) => {
      console.log('✅ WebSocket 连接成功')
      this.isConnected = true
      this.startHeartbeat() // 启动心跳
      this.clearReconnect() // 清除重连
      this.sendCacheMsg()   // 发送缓存消息
    }

    // 接收消息
    this.socket.onmessage = (e) => {
      try {
        const data = JSON.parse(e.data)
        this.dispatch(data) // 分发给订阅者
      } catch (err) {
        this.dispatch(e.data)
      }
    }

    // 关闭连接
    this.socket.onclose = (e) => {
      console.log('❌ WebSocket 断开')
      this.isConnected = false
      this.stopHeartbeat()
      this.reconnect() // 自动重连
    }

    // 错误
    this.socket.onerror = (err) => {
      console.error('❗ WebSocket 异常', err)
      this.isConnected = false
      this.reconnect()
    }
  }

  // ========== 2. 自动重连 ==========
  reconnect() {
    if (!this.reconnectEnabled || this.reconnectTimer) return
    console.log(`🔄 ${this.reconnectDelay / 1000}秒后重连...`)

    this.reconnectTimer = setTimeout(() => {
      this.connect()
      this.reconnectTimer = null
    }, this.reconnectDelay)
  }

  clearReconnect() {
    if (this.reconnectTimer) {
      clearTimeout(this.reconnectTimer)
      this.reconnectTimer = null
    }
  }

  // ========== 3. 心跳保活(最重要!) ==========
  startHeartbeat() {
    if (!this.heartbeatEnabled) return
    this.stopHeartbeat()

    this.heartbeatTimer = setInterval(() => {
      if (this.isConnected) {
        this.send(this.heartbeatMsg)
      }
    }, this.heartbeatInterval)
  }

  stopHeartbeat() {
    if (this.heartbeatTimer) {
      clearInterval(this.heartbeatTimer)
      this.heartbeatTimer = null
    }
  }

  // ========== 4. 发送消息(支持缓存) ==========
  send(data) {
    if (!this.isConnected) {
      this.msgQueue.push(data) // 未连接 → 缓存
      return
    }

    try {
      const msg = typeof data === 'string' ? data : JSON.stringify(data)
      this.socket.send(msg)
    } catch (err) {
      console.error('发送失败', err)
      this.msgQueue.push(data)
    }
  }

  // 发送缓存消息
  sendCacheMsg() {
    while (this.msgQueue.length) {
      const msg = this.msgQueue.shift()
      this.send(msg)
    }
  }

  // ========== 5. 订阅/派发(解耦) ==========
  on(key, callback) {
    this.listeners.set(key, callback)
  }

  off(key) {
    this.listeners.delete(key)
  }

  dispatch(data) {
    this.listeners.forEach((cb) => cb(data))
  }

  // ========== 6. 手动关闭 ==========
  close() {
    this.reconnectEnabled = false // 关闭重连
    this.stopHeartbeat()
    this.clearReconnect()
    this.socket?.close()
    this.isConnected = false
  }
}

// 导出单例
export default new WebSocketClient('ws://localhost:8080/ws')

使用

javascript 复制代码
import ws from '@/utils/websocket'

// 1. 连接
ws.connect()

// 2. 监听消息
ws.on('message', (data) => {
  console.log('收到服务器消息:', data)
})

// 3. 发送消息
ws.send({ type: 'chat', content: '你好' })

// 4. 页面卸载关闭
onUnmounted(() => {
  ws.off('message')
})
相关推荐
故事和你911 小时前
洛谷-【图论2-2】最短路4
开发语言·数据结构·c++·算法·动态规划·图论
摇滚侠1 小时前
14 响应式网页 WEB 前端 WEB 开发 HTML5 + CSS3 + 移动 WEB
前端·css3·html5
vortex51 小时前
Shellinabox 使用指南:基于 Web 的终端模拟器
linux·前端·web ssh
輕華1 小时前
YOLOv10轮毂缺陷检测(下)——模型推理与PyQt5可视化应用
开发语言·qt·yolo
小则又沐风a1 小时前
深入理解进程概念 第三章 进程调度切换
java·linux·服务器·前端
努力攀登的小k1 小时前
《Java基础,Java多态入门到进阶:重写、重载、转型的逻辑与实战避坑》
java·开发语言
问心无愧05131 小时前
ctf show web入门153
笔记
ZC跨境爬虫1 小时前
跟着 MDN 学 HTML day_63:(Web 中矢量图形的完整指南)
前端·javascript·数据库·ui·html
烤麻辣烫1 小时前
计算机思维--经典互联网应用
开发语言·学习·搜索引擎·数据库开发