Arduino+ESP8266+华为云物联网平台实现智能开关

前言

最近在做一个物联网项目,涉及到智能开关的开发。目前已经实现简单的TCP通信远程控制,但是考虑到后期的设备管理以及设备通信所需要的技术和服务器的维护成本,我决定将设备接入云平台。本文将详细阐述如何利用华为云的物联网平台(IoTDA)实现设备的云上管理与通信。

前提条件

准备工作

设备侧选择用MQTT协议,因为对(IoTDA)SDK的编译环境不太了解,这里通过API的方式接入华为云的物联网平台。

获取IoTDA接入信息

首先需要拿到两个信息:

  1. 华为云物联网平台的接入信息,也就是URL和PORT。接入信息通过下图可以看到。

  2. 华为云物联网平台的设备连接鉴权信息。可以参考设备连接鉴权 。简单来讲,就是通过注册设备时的设备ID和密钥生成设备连接鉴权所需的参数(ClientId、Username、Password)。生成参数的链接

安装代码库

其次,设备侧作为MQTT的客户端,需要编写相关代码。基于Arduino IDE开发,可以使用现成的库减少工作量。

  1. 安装PubSubClient库,用于连接和交互MQTT代理服务器(IoTDA)。
  2. 安装ArduinoJson库,用于解析和生成 JSON 数据。

功能实现

基于IoTDA平台实现智能开关,代码要实现以下功能:

  1. 设备进行WiFi连接与网络通信。
  2. 设备通过MQTT协议接入IoTDA平台。
  3. 设备接收IoTDA平台下发的消息。
  4. 设备根据消息执行相应的控制操作。

WiFi连接

c 复制代码
#include <ESP8266WiFi.h>  

const char* ssid = "xxx";   //wifi名称
const char* password = "xxx";  //wifi密码

void setup_wifi() {  
  delay(10);  
  // We start by connecting to WiFi  
  Serial.println();  
  Serial.print("Connecting to ");  
  Serial.println(ssid);  
  
  WiFi.begin(ssid, password);  
  
  while (WiFi.status() != WL_CONNECTED) {  
    delay(500);  
    Serial.print(".");  
  }  
  
  Serial.println("");  
  Serial.println("WiFi connected");  
  Serial.print("IP address: ");  
  Serial.println(WiFi.localIP());  
}

对以上的代码进行解释:

ESP8266WiFi.h 库提供了与 WiFi 网络连接相关的功能,WiFi.begin(ssid, password); 这行代码就可以自动连接WiFi。

接入IoTDA

设备要接入IoTDA,就需要上面准备的接入信息。在代码中定义如下:

c 复制代码
const char* mqttServer = "xxx.myhuaweicloud.com";  
const int mqttPort = 1883;  
const char* mqttClientId = "xxx";      
const char* mqttUser ="xxx";
const char* mqttPassword = "xxx";

#define deviceId "xxx"

#define mqttTopic "$oc/devices/" deviceId "/sys/messages/down"

mqttTopic是在接入成功后,设备侧订阅的Topic,用来接收平台下发的消息。更多的topic定义参考IoTDA平台预置的topic

接下来就需要通过PubSubClient库进行接入IoTDA平台了

c 复制代码
WiFiClient espClient;  
PubSubClient client(espClient);  

void reconnect() {    
  client.setServer(mqttServer, mqttPort);  
  client.setCallback(callback);  
  while (!client.connected()) {  
    Serial.println("Attempting MQTT connection...");  
    if (client.connect(mqttClientId, mqttUser, mqttPassword)) {  
      Serial.println("connected");  
      client.subscribe(mqttTopic);  
    } else {  
      Serial.print(client.state());
      delay(5000);  
    }  
  }  
}  

接收IoTDA消息&控制开关

client.connect 成功后会订阅Topic,当在IoTDA平台下发消息时,就会在 callback 函数进行 digitalWrite(RELAY_PIN, state); 控制开关了。

c 复制代码
void callback(char* topic, byte* payload, unsigned int length) {  
  StaticJsonDocument<128> doc;  
  DeserializationError error = deserializeJson(doc, payload);  
  if (error) {  
    Serial.print(F("deserializeJson() failed: "));  
    Serial.println(error.c_str());  
    return;  
  }  
  
  const char* stateKey = "state";  
  if (doc.containsKey(stateKey)) {  
    bool state = doc[stateKey];  
    digitalWrite(RELAY_PIN, state);  
    Serial.print("Switch state: ");  
    Serial.println(state ? "ON" : "OFF");  
  }  
}  
  

完整代码

c 复制代码
#include <ESP8266WiFi.h>  
#include <PubSubClient.h>  
#include <ArduinoJson.h>  
  
const char* ssid = "xxx";  
const char* password = "xxx.";  
const char* mqttServer = "xxx.myhuaweicloud.com";  
const int mqttPort = 1883;  
const char* mqttClientId = "xxx";      
const char* mqttUser ="xxx";
const char* mqttPassword = "xxx";

#define RELAY_PIN 0


#define deviceId "xxx"

#define mqttTopic "$oc/devices/" deviceId "/sys/messages/down"



WiFiClient espClient;  
PubSubClient client(espClient);  
  
void callback(char* topic, byte* payload, unsigned int length) {  
  StaticJsonDocument<128> doc;  
  DeserializationError error = deserializeJson(doc, payload);  
  if (error) {  
    Serial.print(F("deserializeJson() failed: "));  
    Serial.println(error.c_str());  
    return;  
  }  
  
  const char* stateKey = "state";  
  if (doc.containsKey(stateKey)) {  
    bool state = doc[stateKey];  
    digitalWrite(RELAY_PIN, state);  
    Serial.print("Switch state: ");  
    Serial.println(state ? "ON" : "OFF");  
  }  
}  
  
void reconnect() {  
  while (!client.connected()) {  
    Serial.println("Attempting MQTT connection...");  
    if (client.connect(mqttClientId, mqttUser, mqttPassword)) {  
      Serial.println("connected");  
      client.subscribe(mqttTopic);  
    } else {  
      Serial.print(client.state());
      delay(5000);  
    }  
  }  
}  
  
void setup_wifi() {  
  delay(10);  
  Serial.println();  
  Serial.print("Connecting to ");  
  Serial.println(ssid);  
  
  WiFi.begin(ssid, password);  
  
  while (WiFi.status() != WL_CONNECTED) {  
    delay(500);  
    Serial.print(".");  
  }  
  
  Serial.println("");  
  Serial.println("WiFi connected");  
  Serial.print("IP address: ");  
  Serial.println(WiFi.localIP());  
}  
  
void setup() {  
  Serial.begin(115200);  
  pinMode(RELAY_PIN, OUTPUT); 
  digitalWrite(RELAY_PIN, HIGH); 
  
  setup_wifi();  
  client.setServer(mqttServer, mqttPort);  
  client.setCallback(callback);  
}  
  
void loop() {  
  if (!client.connected()) {  
    reconnect();  
  }  
  client.loop();  
}

IoTDA平台下发消息

将编写完成的代码烧录到ESP-01S模块后,就可以通过IoTDA平台进行消息下发了。消息下发位置见下图

消息下发格式:

json 复制代码
{
    "state":true
}

statetrue时,表示电平状态为HIGH,电路呈开启状态,此时电路不通电。

而当statefalse时,电平状态则为LOW,电路闭合,此时电路处于通电状态。

通过state数值的变化,从而达到远程控制开关。

问题记录

在串口调试时,如果mqtt连接返回-1状态码,意味着连接失败。

那么需要将 PubSubClient.h 库中的两个宏定义MQTT_KEEPALIVEMQTT_MAX_PACKET_SIZE 的数值调大。本文中的代码已分别调整为60和2048。

总结

借助华为云的物联网平台IoTDA,可以帮助物联网行业的用户快速完成设备联网及行业应用集成,大大提高了开发人员的工作效率。而且按需计费每月前一百万条消息是免费的,很大程度上降低了设备上云的成本。

相关推荐
中科岩创8 小时前
广东某海水取排水管线工程边坡自动化监测
大数据·物联网
fan_0008 小时前
LKT4304新一代算法移植加密芯片,守护 物联网设备和云服务安全
物联网·安全
7yewh12 小时前
嵌入式知识点总结 ARM体系与架构 专题提升(四)-编程
arm开发·stm32·单片机·嵌入式硬件·mcu·物联网·51单片机
『往事』&白驹过隙;13 小时前
操作系统(Linux Kernel 0.11&Linux Kernel 0.12)解读整理——内核初始化(main & init)之缓冲区的管理
linux·c语言·数据结构·物联网·操作系统
Jzin13 小时前
【物联网】ARM核常用指令(详解):数据传送、计算、位运算、比较、跳转、内存访问、CPSR/SPSR、流水线及伪指令
arm开发·物联网
老王聊主机16 小时前
2025年华为云一键快速部署幻兽帕鲁联机服务器教程
运维·服务器·华为云
7yewh17 小时前
嵌入式知识点总结 操作系统 专题提升(一)-进程和线程
linux·arm开发·驱动开发·stm32·嵌入式硬件·mcu·物联网
LS·Cui1 天前
第7章 任务的定义与任务切换的实现--总结
物联网
7yewh1 天前
嵌入式知识点总结 C/C++ 专题提升(七)-位操作
c语言·c++·stm32·单片机·mcu·物联网·位操作
laimaxgg2 天前
Linux关于华为云开放端口号后连接失败问题解决
linux·运维·服务器·网络·tcp/ip·华为云