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')
})
相关推荐
CryptoPP3 小时前
快速对接东京证券交易所API数据:实战指南与代码示例
开发语言·人工智能·windows·python·信息可视化·区块链
ZC跨境爬虫3 小时前
跟着 MDN 学JavaScript day_7:数学运算与逻辑判断实战测试
开发语言·前端·javascript·学习·ecmascript
fangdengfu1233 小时前
ES分析系统各个服务日志占用量
java·前端·elasticsearch
Multipath7124 小时前
无人区不掉线:多链路聚合路由,为环塔拉力赛筑起“空中通讯走廊”
网络·5g·安全·无人机·实时音视频
阳区欠4 小时前
【LangChain】LLM基础介绍
开发语言·python·langchain
Jinkxs4 小时前
Java 跨域14-Java 与区块链(Hyperledger)集成
java·开发语言·区块链
JustHappy5 小时前
古法编程秘籍(六):程序到底是怎么跑起来的?从 IO 到中断,一次讲明白
前端·后端·全栈
HYCS5 小时前
用pixi.js实现fabric.js(六):从线性代数的角度理解编辑器交互
前端·javascript·canvas
晨曦中的暮雨5 小时前
Golang速通(Javaer版)
java·开发语言·后端·golang