WebSocket is already in CLOSING or CLOSED state. 报错信息的解决方案

  • 背景
    近期在对 【GatewayWorker】的开发过程中
    注意到,当客户端长时间没有反应时,会发生 WebSocket 自动断开的情况
    在此,提供一个使用定时器的解决方案 …

【分析原因】

【解决步骤】

  • 第一步、首先,根据前面的手册介绍,我在服务端补充了如下的代码:
// 心跳间隔
$gateway->pingInterval = 57;
$gateway->pingNotResponseLimit = 1; // 代表客户端必须定时发送心跳给服务端
$gateway->pingData = '';
  • 第二步、在客户端创建 连接及定时器核心代码如下:
var interval_timer = null;//计时器
var timer_count = 0;
var wsUrl = "wss://xxxeee.com/ssssmn";
var ws;

createOrConnectWebSocket();	// 创建websocket
/**
 * 创建websocket或掉线重连
 */
 function createOrConnectWebSocket(){
    if(!ws){
        //TODO ws 不存在
        ws = new WebSocket(wsUrl);
        websocketInit();
    }else {
        if (!isOnlineCurrUser()){
            ws = null;
            createOrConnectWebSocket();
        }
    }
    // 开启定时器
    init_start_timer();
}

  /**
   * websocket 的初始化
   */
  function websocketInit(){
   ws.onmessage = function (e) {
       var message = eval('(' + e.data + ')');
       switch (message.type) {
           case 'init':
               changeNoReadLogs();
               var bind = '{"type":"bind","from_id":"' + from_id + '","to_id":"' + to_id + '"}';
               ws.send(bind);
               message_load();
               break;
           ....... 具体代码省略......
       }
   };
   ws.onclose = function (e) {
       console.log('websocket 断开: ' + e.code + ' ' + e.reason + ' ' + e.wasClean);
   };
  }
	
    /**
     * 设置一个 30秒的轮询监听方法,避免页面关闭
     */
    function init_start_timer() {
        //重置计数器
        timer_count = 0;
        if(interval_timer != null){
            clearInterval(interval_timer);
            interval_timer = null;
        }
        interval_timer = setInterval(function(){ myTimer() }, 30000);
    }
    /**
     *定时器具体实现方法
     */
    function myTimer() {
        //TODO 如果超过半小时没有交互,则关闭计时器
        if(timer_count >= 1800){
            clearInterval(interval_timer);
        }else {
            timer_count += 30;
            var online = '{"type":"timer","from_id":"' + from_id + '","to_id":"' + to_id + '"}';
            ws.send(online);
            console.log('timer_count',timer_count);
        }
    }
        /**
     * 判断当前用户是否 还在线
     */
    function isOnlineCurrUser() {
        if(ws){
            if(ws.readyState == WebSocket.OPEN){
                return true;
            }else {
                return false;
            }
        }else {
            return false;
        }
    }
  • 然后,注意在发送消息时,
    比如点击发送键时首先判断用户是否在线,如果不在线进行重连或者提示信息

同时,注意当顺利发送或接收到消息时,要进行初始化定时器操作,保证重新计数!

  • 第三步、测试效果 (基本解决了我的需求)
    在发送消息的位置,调用 "createOrConnectWebSocket()" 方法.

【参考文章】

  • 7
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
websocket是一种用于实时通信的协议,它允许浏览器和服务器之间进行双向通信。当出现“websocket is already in closing or closed state”错误时,意味着websocket已经处于关闭状态或正在关闭过程中。 在正常的websocket通信中,两端可以随时通过发送关闭帧来关闭连接。当其中一端发送关闭帧后,另一端也会收到关闭帧,并开始关闭过程。在关闭过程中,双方将交换关闭帧,并逐渐关闭连接。一旦连接完全关闭,websocket处于关闭状态。 但是,如果在关闭过程中尝试发送数据或者出现其他意外情况,websocket可能会进入异常状态,并抛出“websocket is already in closing or closed state”错误。 可能导致此错误的原因有: 1. 在关闭过程中尝试发送数据。一旦关闭帧被发送,websocket连接将处于关闭过程中,此时不应再进行数据发送。 2. 在连接已经关闭的情况下尝试发送数据。一旦连接完全关闭,websocket将无法再发送数据。 3. 实现代码中的错误。可能有一些代码逻辑错误导致了此错误的出现。 为了解决此问题,我们需要进行以下检查和修复措施: 1. 确保在关闭过程中不再尝试发送数据。可以通过代码控制发送数据的时机,避免在连接关闭的状态下发送数据。 2. 检查代码逻辑,确保没有意外地关闭websocket连接,或者在连接已经关闭后继续进行相关操作。 3. 观察错误发生的环境和时间,查找相关代码并进行排查和修复。 总之,当出现“websocket is already in closing or closed state”错误时,我们应该检查代码中的发送数据时机和关闭连接的逻辑,并修复相关问题,以确保websocket通信的正常进行。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值