【ESP8266 QoS 应用】

在本节课里,我们将使用ESP8266配合MQTTfx客户端软件,通过实际操作来看一下MQTT的QoS是如何发挥作用的。

在开始本节课以前请确保您的电脑已经成功安装了MQTTfx软件。另外我们还将使用PubSubClient库。您可以通过以下链接获取此库。

  • 官网地址:https://pubsubclient.knolleary.net/
  • GitHub:https://github.com/knolleary/pubsubclient/
  • 百度网盘下载:https://pan.baidu.com/s/12MHGbdfiOdwOGip5RMSSEQ 提取码: sizy

PubSubClient库目前只支持1级QoS订阅,因此我们将仅介绍如何使用ESP8266通过QoS=1模式订阅和接收MQTT消息。

使用ESP8266接收QoS=1的MQTT消息

要使用QoS=1订阅MQTT消息,需要满足以下要求:

1 接收端要有确定的clientID
2 接收端连接服务器时 cleanSession=False
3 发送端发布消息时 QoS=1或QoS=2
4 接收端订阅消息时 QoS=1

通过以下示例程序,您将可以实现ESP8266 QoS1订阅主题.

本程序旨在演示如何使用PubSubClient库使用ESP8266向MQTT服务器订阅信息。

订阅QoS级别为1。

  • 在此程序控制下,ESP8266启动后将会尝试连接MQTT服务端。

    连接时cleanSession=false。

  • 接下来ESP8266在订阅主题时,将QoS设置为1。

  • 保持ESP8266在线,使用MQTTfx向ESP8266所订阅的主题发布QoS=1的信息。

  • ESP8266将会收到信息(至此还没有体现QoS=1的优势)

  • 将ESP8266断电,然后再次使用MQTTfx向ESP8266订阅主题发布QoS=1的信息。

    此时由于ESP8266未通电,所以无法接收到MQTTfx发送的信息。因此MQTT服务端

    将会保存此信息。

  • 将ESP8266再次通电,ESP8266连接到MQTT服务端后,将会马上收到了MQTTfx在ESP8266断电时所发送的信息。(这就是QoS=1的优势,即客户端断电再通电后依然可以收到QoS=1信息。)

要使用QoS=1订阅MQTT消息,需要满足以下要求:

1 接收端要有确定的clientID

2 接收端连接服务器时 cleanSession=False

3 发送端发布消息时 QoS=1或QoS=2

4 接收端订阅消息时 QoS=1

***********************************************************************/

#include <ESP8266WiFi.h>

#include <PubSubClient.h>

 

// 设置wifi接入信息(请根据您的WiFi信息进行修改)

const char* ssid = "taichi-maker";

const char* password = "12345678";

const char* mqttServer = "test.ranye-iot.net";

// 如以上MQTT服务器无法正常连接,请前往以下页面寻找解决方案

// http://www.taichi-maker.com/public-mqtt-broker/

 

WiFiClient wifiClient;

PubSubClient mqttClient(wifiClient);

 

const int subQoS = 1;     // 客户端订阅主题时使用的QoS级别(截止2020-10-07,仅支持QoS = 1,不支持QoS = 2)

const bool cleanSession = false; // 清除会话(如QoS>0必须要设为false)

 

const char* willTopic = "willTopic"; // 遗嘱主题名称

const char* willMsg = "willMsg";     // 遗嘱主题信息

const int willQos = 0;               // 遗嘱QoS

const int willRetain = false;        // 遗嘱保留

 

void setup() {

  pinMode(LED_BUILTIN, OUTPUT);     // 设置板上LED引脚为输出模式

  digitalWrite(LED_BUILTIN, HIGH);  // 启动后关闭板上LED

  Serial.begin(9600);               // 启动串口通讯

  

  //设置ESP8266工作模式为无线终端模式

  WiFi.mode(WIFI_STA);

  

  // 连接WiFi

  connectWifi();

  

  // 设置MQTT服务器和端口号

  mqttClient.setServer(mqttServer, 1883);

  mqttClient.setCallback(receiveCallback);

 

  // 连接MQTT服务器

  connectMQTTserver();

}

 

void loop() {

  // 如果开发板未能成功连接服务器,则尝试连接服务器

  if (!mqttClient.connected()) {

    connectMQTTserver();

  }

 

   // 处理信息以及心跳

   mqttClient.loop();

  

}

 

// 连接MQTT服务器并订阅信息

void connectMQTTserver(){

  // 根据ESP8266的MAC地址生成客户端ID(避免与其它ESP8266的客户端ID重名)

  String clientId = "client-" + WiFi.macAddress();

 

  /* 连接MQTT服务器

  boolean connect(const char* id, const char* user,

                  const char* pass, const char* willTopic,

                  uint8_t willQos, boolean willRetain,

                  const char* willMessage, boolean cleanSession);

  若让设备在离线时仍然能够让qos1工作,则connect时的cleanSession需要设置为false                

                  */

  if (mqttClient.connect(clientId.c_str(), NULL, NULL, willTopic, willQos, willRetain, willMsg, cleanSession)) {

    Serial.print("MQTT Server Connected. ClientId: ");

    Serial.println(clientId);

    subscribeTopic(); // 订阅指定主题

  } else {

    Serial.print("MQTT Server Connect Failed. Client State:");

    Serial.println(mqttClient.state());

    delay(5000);

  }  

}

 

// 收到信息后的回调函数

void receiveCallback(char* topic, byte* payload, unsigned int length) {

  Serial.print("Message Received [");

  Serial.print(topic);

  Serial.print("] ");

  for (int i = 0; i < length; i++) {

    Serial.print((char)payload[i]);

  }

  Serial.println("");

  Serial.print("Message Length(Bytes) ");

  Serial.println(length);

 

  if ((char)payload[0] == '1') {     // 如果收到的信息以“1”为开始

    digitalWrite(BUILTIN_LED, LOW);  // 则点亮LED。

  } else {                          

    digitalWrite(BUILTIN_LED, HIGH); // 否则熄灭LED。

  }

}

 

// 订阅指定主题

void subscribeTopic(){

  // 建立订阅主题。主题名称以Taichi-Maker-Sub为前缀,后面添加设备的MAC地址。

  // 这么做是为确保不同设备使用同一个MQTT服务器测试消息订阅时,所订阅的主题名称不同

  String topicString = "Taichi-Maker-Sub-" + WiFi.macAddress();

  char subTopic[topicString.length() + 1];  

  strcpy(subTopic, topicString.c_str());

  

  // 通过串口监视器输出是否成功订阅主题以及订阅的主题名称

  // 请注意subscribe函数第二个参数数字为QoS级别。这里为QoS = 1

  if(mqttClient.subscribe(subTopic, subQoS)){

    Serial.print("Subscribed Topic: ");

    Serial.println(subTopic);

  } else {

    Serial.print("Subscribe Fail...");

  }  

}

 

// ESP8266连接wifi

void connectWifi(){

 

  WiFi.begin(ssid, password);


  //等待WiFi连接,成功连接后输出成功信息

  while (WiFi.status() != WL_CONNECTED) {

    delay(1000);

    Serial.print(".");

  }

  Serial.println("");

  Serial.println("WiFi Connected!");  

  Serial.println("");

} 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

暮誠雪

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值