zoukankan      html  css  js  c++  java
  • vue项目中实时通信方式的用法:websocket 与 mqtt

    一、websocket

    在使用websocket的过程中,有时候会遇到网络断开的情况,但是在网络断开的时候服务器端并没有触发onclose的事件。这样会有:服务器会继续向客户端发送多余的链接,并且这些数据还会丢失。所以就需要一种机制来检测客户端和服务端是否处于正常的链接状态。因此就有了websocket的心跳了。还有心跳,说明还活着,没有心跳说明已经挂掉了。

    1. 为什么叫心跳包呢?
    它就像心跳一样每隔固定的时间发一次,来告诉服务器,我还活着。

    2. 心跳机制是?
    心跳机制是每隔一段时间会向服务器发送一个数据包,告诉服务器自己还活着,同时客户端会确认服务器端是否还活着,如果还活着的话,就会回传一个数据包给客户端来确定服务器端也还活着,否则的话,有可能是网络断开连接了。需要重连~

    那么需要怎么去实现它呢?如下所有代码:

    复制代码
    <html>
    <head>
      <meta charset="utf-8">
      <title>WebSocket Demo</title>
    </head>
    <body>
      <script type="text/javascript">
        // var ws = new WebSocket("wss://echo.websocket.org");
        /*
        ws.onerror = function(e) {
          console.log('已关闭');
        };
        ws.onopen = function(e) {
          console.log('握手成功');
          ws.send('123456789');
        }
        ws.onclose = function() {
          console.log('已关闭');
        }
        ws.onmessage = function(e) {
          console.log('收到消息');
          console.log(e);
        }
        */
        
        var lockReconnect = false;//避免重复连接
        var wsUrl = "wss://echo.websocket.org";
        var ws;
        var tt;
        function createWebSocket() {
          try {
            ws = new WebSocket(wsUrl);
            init();
          } catch(e) {
            console.log('catch');
            reconnect(wsUrl);
          }
        }
        function init() {
          ws.onclose = function () {
            console.log('链接关闭');
            reconnect(wsUrl);
          };
          ws.onerror = function() {
            console.log('发生异常了');
            reconnect(wsUrl);
          };
          ws.onopen = function () {
            //心跳检测重置
            heartCheck.start();
          };
          ws.onmessage = function (event) {
            //拿到任何消息都说明当前连接是正常的
            console.log('接收到消息');
            heartCheck.start();
          }
        }
        function reconnect(url) {
          if(lockReconnect) {
            return;
          };
          lockReconnect = true;
          //没连接上会一直重连,设置延迟避免请求过多
          tt && clearTimeout(tt);
          tt = setTimeout(function () {
            createWebSocket(url);
            lockReconnect = false;
          }, 4000);
        }
        //心跳检测
        var heartCheck = {
          timeout: 3000,
          timeoutObj: null,
          serverTimeoutObj: null,
          start: function(){
            console.log('start');
            var self = this;
            this.timeoutObj && clearTimeout(this.timeoutObj);
            this.serverTimeoutObj && clearTimeout(this.serverTimeoutObj);
            this.timeoutObj = setTimeout(function(){
              //这里发送一个心跳,后端收到后,返回一个心跳消息,
              console.log('55555');
              ws.send("123456789");
              self.serverTimeoutObj = setTimeout(function() {
                console.log(111);
                console.log(ws);
                ws.close();
                // createWebSocket();
              }, self.timeout);
    
            }, this.timeout)
          }
        }
        createWebSocket(wsUrl);
      </script>
    </body>
    </html>
    复制代码

    具体的思路如下:
    1. 第一步页面初始化,先调用createWebSocket函数,目的是创建一个websocket的方法:new WebSocket(wsUrl);因此封装成函数内如下代码:

    复制代码
    function createWebSocket() {
      try {
        ws = new WebSocket(wsUrl);
        init();
      } catch(e) {
        console.log('catch');
        reconnect(wsUrl);
      }
    }
    复制代码

    2. 第二步调用init方法,该方法内把一些监听事件封装如下:

    复制代码
    function init() {
      ws.onclose = function () {
        console.log('链接关闭');
        reconnect(wsUrl);
      };
      ws.onerror = function() {
        console.log('发生异常了');
        reconnect(wsUrl);
      };
      ws.onopen = function () {
        //心跳检测重置
        heartCheck.start();
      };
      ws.onmessage = function (event) {
        //拿到任何消息都说明当前连接是正常的
        console.log('接收到消息');
        heartCheck.start();
      }
    }
    复制代码

    3. 如上第二步,当网络断开的时候,会先调用onerror,onclose事件可以监听到,会调用reconnect方法进行重连操作。正常的情况下,是先调用
    onopen方法的,当接收到数据时,会被onmessage事件监听到。

    4. 重连操作 reconnect代码如下:

    复制代码
    var lockReconnect = false;//避免重复连接
    function reconnect(url) {
      if(lockReconnect) {
        return;
      };
      lockReconnect = true;
      //没连接上会一直重连,设置延迟避免请求过多
      tt && clearTimeout(tt);
      tt = setTimeout(function () {
        createWebSocket(url);
        lockReconnect = false;
      }, 4000);
    }
    复制代码

    如上代码,如果网络断开的话,会执行reconnect方法,使用了一个定时器,4秒后会重新创建一个新的websocket链接,重新调用createWebSocket函数,
    重新会执行及发送数据给服务器端。

    5. 最后一步就是实现心跳检测的代码:如下:

    复制代码
    //心跳检测
    var heartCheck = {
      timeout: 3000,
      timeoutObj: null,
      serverTimeoutObj: null,
      start: function(){
        console.log('start');
        var self = this;
        this.timeoutObj && clearTimeout(this.timeoutObj);
        this.serverTimeoutObj && clearTimeout(this.serverTimeoutObj);
        this.timeoutObj = setTimeout(function(){
          //这里发送一个心跳,后端收到后,返回一个心跳消息,
          //onmessage拿到返回的心跳就说明连接正常
          console.log('55555');
          ws.send("123456789");
          self.serverTimeoutObj = setTimeout(function() {
            console.log(111);
            console.log(ws);
            ws.close();
            // createWebSocket();
          }, self.timeout);
    
        }, this.timeout)
      }
    }
    复制代码

    实现心跳检测的思路是:每隔一段固定的时间,向服务器端发送一个ping数据,如果在正常的情况下,服务器会返回一个pong给客户端,如果客户端通过
    onmessage事件能监听到的话,说明请求正常,这里我们使用了一个定时器,每隔3秒的情况下,如果是网络断开的情况下,在指定的时间内服务器端并没有返回心跳响应消息,因此服务器端断开了,因此这个时候我们使用ws.close关闭连接,在一段时间后(在不同的浏览器下,时间是不一样的,firefox响应更快),
    可以通过 onclose事件监听到。因此在onclose事件内,我们可以调用 reconnect事件进行重连操作。

    转载于:https://www.cnblogs.com/tugenhua0707/p/8648044.html

    二、mqtt

    MQTTjs官方地址:MQTT-Github
    在脚手架中使用npm安装到项目:

    npm install mqtt --save

    官方有一个例子:

    var mqtt = require('mqtt')
    var client  = mqtt.connect('mqtt://test.mosquitto.org')
    client.on('connect', function () {
      client.subscribe('presence', function (err) {
        if (!err) {
          client.publish('presence', 'Hello mqtt')
        }
      })
    })
    client.on('message', function (topic, message) {
      // message is Buffer
      console.log(message.toString())
      client.end()
    })
    

    但是这个例子在vue中怎么用并不明显,现在写一下我的例子:

    1.在vue文件中引入mqtt

    import mqtt from 'mqtt'

    2.在data中声明client对象

    data() {
        return {
          mtopic: "1101",
          msg: "",
          client: {}
        };
      }
    

    3. 在mounted钩子中建立连接

    mounted() {
        this.client = mqtt.connect("ws://ip:port", {
          username: "admin",
          password: "password"
        });
        this.client.on("connect", e =>{
          console.log("连接成功");
          this.client.subscribe(this.mtopic, (err) => {
            if (!err) {
              console.log("订阅成功:" + this.mtopic);
            }
          });
        });
        this.client.on("message", (topic, message) => {
          this.msg = message
        });
      }
    

    这里需要注意this指向问题,刚开始的时候因为写的函数所以一直报错,后来把里面的函数换成箭头函数就好了。

    4. 可以把发布消息函数写到methods里面

    methods: {
        handleclick: function() {
          this.client.publish(this.mtopic, this.msg);
        }
      }
    

    最后贴上完整代码

    //vue单文件
    <template>
      <div class="hello">
        <h1>收到的消息:{{msg}}</h1>
        <button @click="handleclick">发布</button>
      </div>
    </template>
    
    <script>
    import mqtt from "mqtt";
    export default {
      name: "HelloWorld",
      data() {
        return {
          mtopic: "1101",
          msg: "lalala",
          client: {}
        };
      },
      mounted() {
        this.client = mqtt.connect("ws://ip:port", {
          username: "admin",
          password: "password"
        });
        this.client.on("connect", e =>{
          console.log("连接成功");
          this.client.subscribe(this.mtopic, (err)=> {
            if (!err) {
              console.log("订阅成功:" + this.mtopic);
            }
          });
        });
        this.client.on("message", (topic, message) => {
          this.msg = message
        });
      },
      methods: {
        handleclick: function() {
          this.client.publish(this.mtopic, this.msg);
        }
      }
    };
    </script>

    转载于:https://blog.csdn.net/m0_37811924/article/details/102602914

  • 相关阅读:
    What is PE ?
    指针和引用(int*、int&、int*&、int&*、int**)
    诺顿12 免许可 英文版 Symantec Endpoint Protection 12.1.671.4971 下载
    RadControls for ASP.NET AJAX Q2 2011 最新版下载+源码下载
    请大家给小弟 改个SQL的错 谢谢啦!!
    [置顶]DAEMON Tools Pro Advanced 4.41.0314.0232 破解版
    Setting Environment Variable PATH on Ubuntu
    小弟的程序有点问题 请高手帮忙改一下
    Session 有没有必要使用它
    ASP.NET 2.0加密Web.config 配置文件
  • 原文地址:https://www.cnblogs.com/dancer0321/p/13993979.html
Copyright © 2011-2022 走看看