1、在utils下新建websocket.js文件

var websock = null
let rec //断线重连后,延迟5秒重新创建WebSocket连接  rec用来存储延迟请求的代码
let isConnect = false //连接标识 避免重复连接
let checkMsg = 'heartbeat' //心跳发送/返回的信息 服务器和客户端收到的信息内容如果如下 就识别为心跳信息 不要做业务处理
const wsUrl = 'ws://localhost:8888'
var globalCallback = function () {}

let createWebSocket = () => {
  try {
    websock = new WebSocket(wsUrl)
    initWebSocket() //初始化websocket连接
  } catch (e) {
    console.log('尝试创建连接失败')
    reConnect() //如果无法连接上webSocket 那么重新连接!可能会因为服务器重新部署,或者短暂断网等导致无法创建连接
  }
}

//定义重连函数
let reConnect = () => {
  console.log('尝试重新连接')
  if (isConnect) return //如果已经连上就不在重连了
  rec && clearTimeout(rec)
  rec = setTimeout(function () {
    // 延迟5秒重连  避免过多次过频繁请求重连
    createWebSocket()
  }, 5000)
}
//设置关闭连接
let closeWebSocket = () => {
  webSocket.close()
}
//心跳设置
var heartCheck = {
  timeout: 2000, //每段时间发送一次心跳包 这里设置为20s
  timeoutObj: null, //延时发送消息对象(启动心跳新建这个对象,收到消息后重置对象)

  start: function () {
    this.timeoutObj = setInterval(function () {
      if (isConnect) {
        console.log('心跳检测向服务端发送消息:' + checkMsg)
        websock.send(checkMsg)
      } else {
        // 重连
        console.log('尝试重连:' + checkMsg)
        reConnect()
      }
    }, this.timeout)
  },

  reset: function () {
    clearTimeout(this.timeoutObj)
    this.start()
  },
}

// 初始化websocket
function initWebSocket() {
  // ws地址 -->这里是你的请求路径
  let uName = localStorage.getItem('uName')
  websock = new WebSocket(wsUrl)
  websock.onmessage = function (e) {
    websocketonmessage(e)
  }
  websock.onclose = function (e) {
    websocketclose(e)
  }
  websock.onopen = function () {
    websocketOpen()
    // 开启心跳检测方法
    heartCheck.start()
  }

  // 连接发生错误的回调方法
  websock.onerror = function () {
    console.log('WebSocket连接发生错误')
    isConnect = false //连接断开修改标识
    reConnect() //连接错误 需要重连
  }
}

// 实际调用的方法
function sendSock(agentData, callback) {
  globalCallback = callback
  // console.log(globalCallback)
  if (websock.readyState === websock.OPEN) {
    // 若是ws开启状态
    websocketsend(agentData)
  } else if (websock.readyState === websock.CONNECTING) {
    // 若是 正在开启状态,则等待1s后重新调用
    setTimeout(function () {
      sendSock(agentData, callback)
    }, 1000)
  } else {
    // 若未开启 ,则等待1s后重新调用
    setTimeout(function () {
      sendSock(agentData, callback)
    }, 1000)
  }
}

function getSock(callback) {
  globalCallback = callback
}
// 数据接收
function websocketonmessage(e) {
  // 接收到服务端消息
  console.log('接收到服务端消息:' + e.data)
  let result = JSON.stringify(e.data)

  if (!result) {
    heartCheck.reset()
  } else {
    if (result.msg == 'open success') {
      sessionStorage.setItem('wid', result.wid)
    } else {
      console.log('准备发给前端的消息:' + result)
      if (!globalCallback) return
      globalCallback(result)
    }
  }

  // globalCallback(JSON.parse(e.data))
  function decodeUnicode(str) {
    str = str.replace(/\\/g, '%')
    //转换中文
    str = unescape(str)
    //将其他受影响的转换回原来
    str = str.replace(/%/g, '\\')
    //对网址的链接进行处理
    str = str.replace(/\\/g, '')
    return str
  }
}

// 数据发送
function websocketsend(agentData) {
  console.log('发送给服务端:' + JSON.stringify(agentData))
  websock.send(JSON.stringify(agentData))
}

// 关闭
function websocketclose(e) {
  console.log(e)
  isConnect = false //断开后修改标识
  console.log('connection closed (' + e.code + ')')
}

// 创建 websocket 连接
function websocketOpen(e) {
  isConnect = true
  console.log('连接成功')
}

initWebSocket()

// 将方法暴露出去
export { sendSock, getSock, createWebSocket, closeWebSocket }

2、挂载到main.js下

// WebSocket封装方法
import * as socketApi from './utils/socket'
Vue.prototype.socketApi = socketApi

3、在组件中使用

image-1653977740395

版权声明:本文为CSDN博主「HarryHY」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/HarryHY/article/details/100656030