一、背景

        Websocket是html5提出的一个协议规范,是为解决客户端与服务端实时通信。
        WebSocket在连接关闭时会触发onclose事件, 在连接异常时会触发onerror事件。但在弱网环境下, 它们的触发灵敏度不高, 往往延迟很久才触发, 前端再去进行重连操作, 这样很不友好。

        本文将介绍使用心跳重连机制来改善这一问题。

二、原理

        心跳重连机制:前端在WS连接成功的情况下,开始执行心跳函数,首先向服务器端发送ping信息, 服务器内若收到信息则会返回pong信息。在一定时间内, 前端收到服务器返回的信息, 则表示此连接是正常的, 便重置心跳函数; 若前端在一定时间内没有收到心跳函数, 则表明没有连接成功, 此时前端关闭ws, 再执行重连操作。

三、代码

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>

    <style>
      #container {
        display: grid;
        /*三列, 每列宽50px*/
        grid-template-columns: repeat(4, 50px);
        /*三行, 每行高50px*/
        grid-template-rows: 50px 50px 50px;
        margin-top: 20px;
      }

      .item {
        border: 1px solid #e5e4e9;
        cursor: pointer;
        font-size: 12px;
        color: #ffffff;
        display: flex;
        justify-content: center;
        align-items: center;
      }

      .item-1 {
        background-color: #ef342a;
      }
    </style>
  </head>

  <body>
    <h1>Websocket心跳检测</h1>
    <div id="container">
      <div id="btn" class="item item-1">手动停止</div>
    </div>
    <script>
      class Socket {
        constructor(options) {
          // Websocket地址
          this.url = options.url;
          // 监听服务器返回数据
          this.handleMessage = options.handleMessage;
          // Websocket实例对象
          this.ws = null;
          // 心跳函数定时器1:定时发送指令
          this.pingTimer = null;
          // 心跳函数定时器2:定时关闭重连
          this.serverTimer = null;
          this.timeout = options.timeout;
        }

        /**
         * 建立连接
         */
        connect() {
          this.ws = new WebSocket(this.url);

          this.ws.onopen = (e) => {
            console.log("连接成功", e);
            // 启动心跳检测
            this.handleHeartCheck();
          };

          this.ws.onmessage = (e) => {
            this.handleMessage(e.data);
            // 接收到消息则重置心跳函数
            this.handleHeartCheck();
          };

          this.ws.onclose = (e) => {
            console.log("连接关闭", e);
            this.handleClose(e);
          };

          this.ws.onerror = (e) => {
            console.log("连接报错", e);
            this.handleClose(e);
          };
        }

        /**
         * 心跳检查
         */
        handleHeartCheck() {
          if (this.pingTimer) {
            clearTimeout(this.pingTimer);
            this.pingTimer = null;
          }
          if (this.serverTimer) {
            clearTimeout(this.serverTimer);
            this.serverTimer = null;
          }
          this.pingTimer = setTimeout(() => {
            // 发送探测指令
            this.ws.send(JSON.stringify({ type: "ping" }));
            // 超过一定时间还没响应, 手动关闭,再重连
            this.serverTimer = setTimeout(() => {
              console.log("断线重连");
              this.ws.close();
            }, this.timeout);
          }, this.timeout);
        }

        /**
         * 关闭webSocket
         */
        handleClose(e) {
          if (this.pingTimer) {
            clearTimeout(this.pingTimer);
            this.pingTimer = null;
          }
          if (this.serverTimer) {
            clearTimeout(this.serverTimer);
            this.serverTimer = null;
          }
          this.connect();
        }
      }
    </script>
    <script>
      // 初始化
      const ws = new Socket({
        url: "ws://localhost:8765",
        timeout: 3000, // 心跳检测时间
        handleMessage: function (data) {
          // 监听服务器返回信息
          console.log("服务器返回信息", data);
        },
      });
      ws.connect();
      document.getElementById("btn").onclick = () => {
        console.log("手动关闭");
        ws.ws.close();
      };
    </script>
  </body>
</html>

四、Stomp介绍

        概念:面向消息的简单文本协议

        websocket定义了两种传输信息类型:文本信息和二进制信息。类型虽然被确定,但是他们的传输体是没有规定的。所以,需要用一种简单的文本传输类型来规定传输内容,它可以作为通讯中的文本传输协议。

        STOMP是基于帧的协议,客户端和服务器使用STOMP帧流通讯。

        一个STOMP客户端是一个可以以两种模式运行的用户代理,可能是同时运行两种模式。

        作为生产者,通过SEND框架将消息发送给服务器的某个服务。

        作为消费者,通过SUBSCRIBE制定一个目标服务,通过MESSAGE框架,从服务器接收消息。

五、Stomp常用代码

// 发送消息
stompClient.send('/topic/terminal_chart', {}, "ping") 
// 订阅(subscription是一个对象,包含一个id属性,对应这个这个客户端的订阅ID)
var subscription = client.subscribe("/queue/test", callback);
// 终止订阅消息
subscription.unsubscribe();
// 获取STOMP子协议的客户端对象
stompClient = Stomp.over(socket);
// 心跳发送频率
stompClient.heartbeat.outgoing = 20000;
// 心跳接收频率
stompClient.heartbeat.incoming = 20000;
// 调用.connect方法连接Stomp服务端进行验证
stompClient.connect({}, ()=>{
    // 成功回调
}, ()=>{
    // 失败回调(第一次连接失败和连接后断开连接都会调用这个函数)
})
// 判断是否连接
stompClient.connected

六、Stomp自带心跳机制

stomp默认的心跳为10000ms,
heartbeat.outgoing:客户端发给服务端的心跳,* 0表示它不能发送心跳 * 否则它是能保证两次心跳的最小毫秒数
heartbeat.incoming:客户端希望服务端发送的心跳。* 0表示它不想接收心跳 * 否则它表示两次心跳期望的毫秒数
CONNECT
heart-beat:<cx>,<cy> 客户端

CONNECTED:
heart-beat:<sx>,<sy> 服务端
对于client发送server的心跳: * 如果<cx>为0(client不能发送心跳)或者<sy>为0(server不想接收心跳),将不起任何作用。心跳频率为MAX(<cx>,<sy>)毫秒数.
对于server发送client的心跳:心跳频率为MAX(<cy>,<sx>)毫秒数.

七、Stomp手写心跳检测

        如果服务端禁止接收心跳,则stomp自带心跳机制将不起作用,如下手写的代替。

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
    <script src="./stomp.js"></script>
  </head>
  <body>
    <script>
      class Socket {
        constructor(options) {
          // 地址
          this.url = options.url;
          // 目的地
          this.target = options.target;
          // 客户端对象
          this.stompClient = null;
          // 心跳频率
          this.timeout = options.timeout;
          // 定时发送心跳
          this.pingTimeout = null;
          // 定时重连
          this.serverTimer = null;
          // 服务端返回的数据处理
          this.handleMessage = options.handleMessage;
        }
        connect() {
          let websocket = new WebSocket(this.url);
          this.stompClient = Stomp.over(websocket);
          this.stompClient.connect(
            {},
            this.connectCallback,
            this.errorCallback
          );
        }
        // 成功回调
        connectCallback = (e) => {
          console.log("连接成功", e);
          this.handleHeartBeat();
          this.stompClient.subscribe(this.target, (res) => {
            if (res.body === "ping") {
              console.log("心跳返回数据", res);
              this.handleHeartBeat();
            } else {
              // 其他逻辑
              this.handleMessage(res);
            }
          });
        };
        // 失败回调(连接失败时的回调函数,此函数重新调用连接方法,形成循环,直到连接成功)
        errorCallback = () => {
          console.log("失败重连");
          if (this.pingTimeout) {
            clearInterval(this.pingTimeout);
            this.pingTimeout = null;
          }
          if (this.serverTimer) {
            clearInterval(this.serverTimer);
            this.serverTimer = null;
          }
          this.connect();
        };

        handleHeartBeat() {
          if (this.pingTimeout) {
            clearInterval(this.pingTimeout);
            this.pingTimeout = null;
          }
          if (this.serverTimer) {
            clearInterval(this.serverTimer);
            this.serverTimer = null;
          }
          this.pingTimeout = setTimeout(() => {
            // 发出探测指令
            this.stompClient.send(this.target, {}, "ping");
            this.serverTimer = setTimeout(() => {
              this.errorCallback();
            }, this.timeout);
          }, this.timeout);
        }
      }
    </script>
    <script>
      let ws = new Socket({
        url: "ws://10.100.31.48:1995/customer/bj-metro-server-customer",
        target: "/topic/terminal_chart",
        timeout: 10000,
        // 订阅信息的返回
        handleMessage: (res) => {},
      });
      ws.connect();
    </script>
  </body>
</html>