类组件化websocket的方法(心跳机制)

/**
 * WebSocket统一管理
 */
export class WebSocketClient {
    constructor(url) {
        if (!url) {
            throw new Error("WebSocket URL is required.");
        }
        this.url = url;
        this.websocket = null;
        this.listeners = {};
        this.heartbeatInterval = 30000; // 心跳检测间隔(毫秒)
        this.reconnectDelay = 10000; // 断线后重连间隔(毫秒)
        this.pingTimeout = null;
        this.reconnectTimeout = null;
        this.isManuallyClosed = false;
    }

    /** 初始化 WebSocket 连接 */

    connect() {
        this.websocket = new WebSocket(this.url);

        this.websocket.onopen = () => {
            // console.log("WebSocket connection opened.");
            // this.startHeartbeat(); // 开始心跳检测
            this.dispatch("open");
        };

        this.websocket.onmessage = (event) => {
            const data = JSON.parse(event.data ?? "{}");
            this.dispatch("message", data);
        };

        this.websocket.onclose = () => {
            // console.log("WebSocket connection closed.");
            this.dispatch("close");
            // this.stopHeartbeat(); // 停止心跳检测
            if (!this.isManuallyClosed) {
                this.reconnect(); // 自动重连
            }
        };

        this.websocket.onerror = (error) => {
            console.error("WebSocket error:", error);
            this.dispatch("error", error);
        };
    }

    // 关闭 WebSocket 连接
    close() {
        this.isManuallyClosed = true; // 手动关闭连接时,不进行重连
        if (this.websocket) {
            this.websocket.close();
        }
    }

    /**
     * 添加事件监听函数
     * @param { 'open' | 'message' | 'close' | 'error' } eventType
     * @param { (data) => void } callback
     */
    addListener(eventType, callback) {
        if (!this.listeners[eventType]) {
            this.listeners[eventType] = [];
        }
        this.listeners[eventType].push(callback);
    }

    /** 移除事件监听函数 */
    removeListener(eventType, callback) {
        if (!this.listeners[eventType]) return;

        this.listeners[eventType] = this.listeners[eventType].filter(
            (listener) => listener !== callback
        );
    }

    /** 派发事件 */
    dispatch(eventType, data) {
        if (!this.listeners[eventType]) return;

        this.listeners[eventType].forEach((listener) => listener(data));
    }

    // 心跳检测 (ping/pong)
    startHeartbeat() {
        if (this.pingTimeout) {
            clearTimeout(this.pingTimeout);
        }

        // 定时发送心跳
        this.pingTimeout = setTimeout(() => {
            if (this.websocket && this.websocket.readyState === WebSocket.OPEN) {
                this.websocket.send("ping"); // 发送心跳包
                console.log("Ping sent to server.");
            }

            // 继续心跳
            this.startHeartbeat();
        }, this.heartbeatInterval);
    }

    // 停止心跳检测
    stopHeartbeat() {
        if (this.pingTimeout) {
            clearTimeout(this.pingTimeout);
        }
    }

    // 自动重连
    reconnect() {
        console.log(`正在重连${this.url} ${this.reconnectDelay / 1000} 秒...`);
        this.reconnectTimeout = setTimeout(() => {
            this.connect();
        }, this.reconnectDelay);
    }

    send(data) {
        this.websocket.send(data);
    }
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我理解你的问题是关于使用 `setup` 语法糖封装 WebSocket心跳机制方法。首先, `setup` 是 Vue 3 中引入的新特性,它可以在组件实例之前执行一些操作,比如初始状态、订阅事件、创建 WebSocket 连接等。下面是一个使用 `setup` 语法糖封装 WebSocket心跳机制的示例代码: ```vue <template> <div>{{ message }}</div> </template> <script> import { ref, onMounted, onUnmounted } from 'vue' export default { setup() { const message = ref('') let socket = null const connectWebSocket = () => { socket = new WebSocket('wss://example.com/ws') socket.onopen = () => { console.log('WebSocket connected') socket.send('Hello, WebSocket') } socket.onmessage = event => { message.value = event.data } socket.onclose = () => { console.log('WebSocket disconnected') setTimeout(() => { connectWebSocket() }, 5000) } } const startHeartbeat = () => { setInterval(() => { if (socket.readyState === WebSocket.OPEN) { socket.send('Heartbeat') } }, 3000) } onMounted(() => { connectWebSocket() startHeartbeat() }) onUnmounted(() => { socket.close() }) return { message } } } </script> ``` 上面的代码中,我们使用了 `ref` 创建了一个响应式的 `message` 变量,将其绑定到模板中的 `<div>` 元素上。在 `setup` 函数中,我们声明了一个 `socket` 变量,用于存储 WebSocket 连接对象。然后,我们定义了 `connectWebSocket` 函数,用于创建 WebSocket 连接,并在连接打开、收到消息、连接关闭时分别执行对应的操作。接着,我们定义了 `startHeartbeat` 函数,用于定时发送心跳包。在 `onMounted` 生命周期钩子中,我们调用了 `connectWebSocket` 和 `startHeartbeat` 函数。在 `onUnmounted` 生命周期钩子中,我们关闭了 WebSocket 连接。最后,我们将 `message` 变量返回给模板使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值