WebSocket连接MQTT方案之阿里云及其它平台通解

WebSocket连接MQTT方案之阿里云及其它平台通解

前言:前段时间突发奇想,做一个物联网项目。下位机通过ESP8266连接云平台(oneNET)将数据传给WEB端(VUE写的)。于是问题就发生了。。。web端能够连接EMQX的mqtt,并且成功订阅和通讯,但是就是连接不上OneNET的MQTT。于是查找了几乎网上所有现有的WEB连接mqtt的方案,没有一个成功。而此时我还是在怀疑是自己的代码问题(因为此前也做过oneNET的mqtt连接,只不过是android端的而已),按道理只是将连接时的参数改成云平台的对应的就行的。最后去看了oneNET的官方文档,发现不支持Websocket!!!(之前也看过,但好像忽略了这一点)。于是在心态接近崩溃的边缘,改用了连接阿里云的方案。。。

环境:

    1. Vue:vue3

    2. Vue-cli: 4.5.15

    3. IDE: Vs code

    4. Mqtt:4.3.7


阿里云的物联网云服务器怎么申请,创建设备这些我就不说了,一搜一大堆。只要记得拿到连接参数。

实现方案

由于我的界面有子界面,并且使用mqtt就是要实时监控设备的状态以及接收数据和下发指令。如果没切换一个页面就要去重新连接一遍,那我为什么不用HTTP呢。所以如果只是要简单的学习,可以写在需要连接的界面的生命周期函数里去实现的。如果想要解决切换重连的问题,则需要mqtt的连接实例一直存在,且外部变化,如界面刷新不能轻易的改变mqtt连接状态,而解决这个问题就可以用到Vuex去解决。vuex可用于父子界面传值,那么我在跳转界面时,在创建时则获取到连接实例,并且赋值一个新的变量,不就能监听和发送服务器给的消息了吗。

由于用的Vue3的写法,可能和2的写法有很大差别,但只要看懂了,应该还是很容易重现的。

代码实现

目录:index.js  (vuex)

// 导入

import{createStore}from'vuex'

importmqttfrom'mqtt'

// 连接参数

varoptions={

// 下面这些参数在每个平台的叫法都不一样  阿里云,设备里有个mqtt连接参数,名字就是这些

connectTimeout:4000,// 超时时间

clientId:'XXXXXXXXXX',//在oneNET中这个是产品ID

username:'XXXXXXXXX',// 在oneNET中这个是鉴权设备ID

password:'XXXXXXXXX',//在oneNET中这个是鉴权

cleanSession:false,// 保留信息

keepAlive:60// 心跳值

}

// 要订阅的主题   每个平台的规则也不一样,一般来说只要订阅同一个主题就可以了,主题名字都可以随意的。

varsubscription={

topic:'XXXXXX',// 主题名字

// 通讯质量,有0,1,2。

//0是至多发一次,如果要接收的客服端不在线,则消息可能丢失

//1是至少发一次,如果对方没接到,会重发,但无法保障消息重复

//2是对1的限制,会重发,但是会保障消息到达对方并且只会到达一次

qos:0

}

//VueX

exportdefaultcreateStore({

state:{

client:null,//连接

recvData:""//接收消息

   },

mutations:{//可以写入方法(非异步)

//连接mqtt

MqttClient(state){

try{

//创建连接对象   其中 xxxx 为IP地址或者域名(阿里云用的是域名),xx为端口号,options为连接参数(上面已经定义)

state.client=mqtt.connet("ws://xxxx:xx",options)

}catch(error){

console.log("mqtt连接失败",error)

           }

//连接成功以后,订阅主题

state.client.on("connect",(error)=>{

console.log("连接成功")

    const{topic,qos}=subscription//获取上面对象里的两个值

try{

state.client.subscribe(topic,qos, (error)=>{//订阅主题

if(!error) {

console.log('订阅成功')

}else{

console.log('订阅失败')

                        }

  })

}catch(error) {

console.log('订阅错误:',error)

               }

           })

       }

   },

actions:{//可以实现异步方法

connectMqtt(context) {

      context.commit("MqttClient")//为什么要再封装呢,而不直接使用上面那个。因为连接类最好在可以使用异步调用的地方。

    }

   },

modules{}

})

至此,vuex里的代码就写完了。注意:阿里云订阅主题的是有规则的,不像其它平台可以直接用主题名就行了。去产品里,点击查看,进去以后点击Topic类列表,找到基础通信Topic,找到广播那一栏。然后根据文档正确使用。如我要订阅一个名叫“test”的主题,其它平台可能直接就输入“test”就可以订阅了,但是阿里云要按照文档规则输入  "/broadcast/XXXXXX/test"  才能正常使用,否则就算订阅了也无法直接通过订阅主题然后完成两个设备通讯。并且连接参数给的端口号为1883,但websocket不能使用这个,要使用443。而ESP8266的下位机要用1883才行。

目录:Home.vue

<template>

    <divid="main">

     ------界面我就不展示代码了--------

</div>

</template>

<scriptsetup>

import{useStore}from"vuex";

//这里直接使用了setup,这是vue3特有的,就不需要再去实现create这些了,但还是写

//export default{

//setup(){

//return {}

//}

//}

// 在script 后面申明了以后就不用写了。

conststore=useStore()//创建一个vuex的实例

//连接mqtt  这个函数原本要在最外层(父级)实现的,这个函数必须被调用一次,但不能重复调用,否则会照成多连接或者重连。

//在这里调用是因为讲解。

store.dispatch("connectMqtt");

constclient//创建一个连接变量,并赋值给他mqtt的实例连接,方便后面使用mqtt的回调

//可以看做一个监听回调,只要接到消息都会走一遍这个方法的。 其中topic为主题名字,message为接到的内容。

//如果要实现发送那些功能,可以去看下其它的博客。其实也只是调用对应的方法而已。

client.on("message",(topic,message)=>{

console.log('接到来自主题'+topic+"的信息:"+message)

   })

</script>

到此就完成了连接、订阅、接收

最后

这里我只写了连接阿里云一种,为什么标题会是通解呢。其实稍微了解过mqtt的人就会知道,其实它不过是一个协议罢了,要实现很简单的,别人已经为我们封装好了而已。我们只管实现封装好的方法就行了。而只要支持mqtt协议的平台,基本都可以通过不改变实现方法,改变连接参数就可以连接了(这里再吐槽一下oneNET,不支持websocket!!!)。我写的app当时是为了连接oneNET而写的,但只要改变IP地址、端口号、设备ID、产品ID、密码(鉴权)一样能够连接阿里云的物联网平台和EMQX。而这里用Vue3,用vuex去实现连接,其实只要看懂了方法,用vue2,在created(){}中去实现连接一样的可以的(只是刷新要重连罢了)。

注:转载请标明出处!

csdn:WebSocket连接MQTT方案之阿里云及其它平台通解_Dream's的博客-CSDN博客

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容