当前位置: 首页 > article >正文

Vue3_开启全局websocket

1、封装websocket

新建文件夹"socket.ts",路径:"@/utils/socket"

export default (onMessage: Function) => {
  let socketUrl = 'ws://171.29.8.218:8080/ems/ws/screen' //socket请求地址
  let socket: WebSocket
  let lockReconnect = false
  let timer: ReturnType<typeof setTimeout>
  const createSocket = () => {
    try {
      socket = new WebSocket(socketUrl)
      init()
    } catch (e) {
      console.log('catch' + e)
      reconnect()
    }
  }
  const reconnect = () => {
    if (lockReconnect) return
    lockReconnect = true
    clearTimeout(timer)
    timer = setTimeout(() => {
      createSocket()
    }, 4000)
  }

  const init = () => {
    socket.onopen = function (event) {
      console.log('WebSocket:已连接')
      //心跳检测重置
      heartCheck.reset().start()
    }

    //接收到消息的回调方法
    socket.onmessage = function (event) {
      console.log('WebSocket:收到一条消息', event.data)
      const isHeart = /pong/.test(event.data)
      if (onMessage && !isHeart) {
        //触发自定义onMessage
        onMessage.call(null, event)
      }
      heartCheck.reset().start()
    }

    //连接发生错误的回调方法
    socket.onerror = function (event) {
      console.log('WebSocket:发生错误')
      reconnect()
    }

    //连接关闭的回调方法
    socket.onclose = function (event) {
      console.log('WebSocket:已关闭')
      heartCheck.reset() //心跳检测
      reconnect()
    }

    //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常
    window.onbeforeunload = function () {
      socket.close()
    }
  }

  var heartCheck = {
    timeout: 8000,
    timeoutObj: setTimeout(() => {}),
    serverTimeoutObj: setInterval(() => {}),
    reset: function () {
      clearTimeout(this.timeoutObj)
      clearTimeout(this.serverTimeoutObj)
      return this
    },
    start: function () {
      var self = this
      clearTimeout(this.timeoutObj)
      clearTimeout(this.serverTimeoutObj)
      this.timeoutObj = setTimeout(function () {
        //这里发送一个心跳,后端收到后,返回一个心跳消息,
        //onmessage拿到返回的心跳就说明连接正常
        socket.send(
          JSON.stringify({
            action: 'ping'
          })
        )
        self.serverTimeoutObj = setTimeout(function () {
          //如果超过一定时间还没重置,说明后端主动断开了(因此timeout设定时间要比后端返回socket间隔时间长,否则会判定后端主动断开)
          console.log('关闭服务')
          socket.close() //如果onclose会执行reconnect,我们执行 websocket.close()就行了.如果直接执行 reconnect 会触发onclose导致重连两次
        }, self.timeout)
      }, this.timeout)
    }
  }
  createSocket()
}

2、全局注册

在文件夹"main.ts"中

import Websocket from "./utils/socket"
const onMessageList: Array<Function> = []; // 声明一个全局变量来收集onMessage列表,因为在不同的页面可能会有不同的处理
app.provide("onMessageList", onMessageList); // 全局注入
const onMessage = (event: any) => {
	// 遍历onMessage集合并触发
    onMessageList.forEach(f => {
        f.call(null, event);
    })
}
Websocket(onMessage); // 启动websocket

3、页面使用

import { inject } from "vue";
const onMessageList = inject("onMessageList"); // 接收注入
const socketMessage = (res) => {
    const data = JSON.parse(res.data); // 获取数据
};
onMessageList.push(socketMessage); // 将自定义处理事件插入onMessge集合

http://www.kler.cn/a/369223.html

相关文章:

  • 「Java SPI机制应用快速入门」: 一种JDK内置的服务提供发现机制
  • 【C++】继承与模板
  • Ceph 存储系统全解
  • 【进阶】面向对象之接口(多学三招)
  • 微信小程序的日期区间选择组件的封装和使用
  • Python之Excel自动化处理(三)
  • Qt6切换音轨
  • ffmpeg视频滤镜:均值模糊-boxblur
  • MAN Truck Bus EDI 需求分析
  • Flutter Column组件实战案例
  • 2024 最新 frida技术栈 第一部分
  • Linux云服务器安装Docker、MySQL、Redis
  • 国产系统安装Oracle报错处理
  • 利用 Google AI 工具提升应用智能化:ML Kit、TensorFlowLite、Cloud Vision、AutoML、Gemini
  • 手机折叠屏贴膜应用
  • 【AI日记】24.10.27 了解AI的未来
  • 0基础学java之Day16
  • 缓存预取文章比较分析
  • 来源爬虫程序调研报告
  • 【山西】《信息化项目软件运维费用测算指南》(DB 14/T 2163-2020)-省市费用标准解读系列01
  • 【Vue】word / excel / ppt / pdf / 视频(mp4,mov) 预览
  • 「ZJUBCA秋季迎新见面会预告」
  • Netty的简介与实战
  • Java运行时数据区
  • 助力AI智能化时代:全国产化飞腾FT2000+/64+昇腾310B服务器主板
  • 关于k8s的cilium网络插件踩坑记