当前位置: 首页 > article >正文

通过mqtt 点灯

1 解析mqtt 传过来的json

        用cjson 解析。

2   类似mvc的结构,调用具体的动作函数

  1. 定义设备处理结构体:使用结构体数组映射设备名称与处理函数,实现可扩展的指令分发
  2. 分离设备逻辑:为每个设备(如 LED、Motor)编写独立的处理函数,保持模块解耦
  3. 统一 JSON 解析流程:在解析函数中通过遍历结构体数组匹配指令,调用对应处理函数
  4. 状态校验机制:对输入的状态值进行合法性校验,避免无效操作

完整修改后代码

#include "system.h"
#include "SysTick.h"
#include "led.h"         // 添加LED函数声明头文件
#include "usart.h"
#include "wifi_config.h"
#include "wifi_function.h"
#include <string.h>
#include "cJSON.h"// 声明外部变量
extern struct STRUCT_USARTx_Fram strEsp8266_Fram_Record;// 函数声明
void ProcessWiFiData(uint8_t *data, uint16_t length);
void HandleMQTTMessage(const char *topic, const char *message, const char *message_le);// 全局变量
char topic_id[20] = "topic911";
char sub_topic_id[20] = "topic108";
uint8_t wifi_rx_buffer[RX_BUF_MAX_LEN]; // 用于暂存接收数据的缓冲区// ========== 设备处理结构体定义 ==========
typedef enum {DEVICE_LED,DEVICE_MOTOR,DEVICE_UNKNOWN
} DeviceType;typedef struct {const char *device_name;        // 设备名称(JSON键名)DeviceType device_type;         // 设备类型枚举void (*handler)(const char *state); // 处理函数指针
} DeviceHandler;// ========== 设备处理函数实现 ==========
// LED处理函数(假设LED_Init()已在led.h中声明)
void LedHandler(const char *state) {if (strcmp(state, "on") == 0) {LED_ON(); // 假设LED_ON()在led.c中实现printf("Led turned on\r\n");} else if (strcmp(state, "off") == 0) {LED_OFF(); // 假设LED_OFF()在led.c中实现printf("Led turned off\r\n");} else {printf("Invalid state for LED: %s\r\n", state);}
}// Motor处理函数(需根据实际硬件实现)
void MotorHandler(const char *state) {if (strcmp(state, "on") == 0) {// 这里添加电机启动代码printf("Motor started\r\n");} else if (strcmp(state, "off") == 0) {// 这里添加电机关闭代码printf("Motor stopped\r\n");} else {printf("Invalid state for Motor: %s\r\n", state);}
}// 设备处理注册表(可扩展)
DeviceHandler device_handlers[] = {{"Led", DEVICE_LED, LedHandler},{"Motor", DEVICE_MOTOR, MotorHandler},{NULL, DEVICE_UNKNOWN, NULL} // 结束标志
};// ========== JSON解析函数扩展 ==========
void parse_json_message(const char *message) {cJSON *root;cJSON *item; // 修正变量名,避免未使用警告if (message == NULL) {printf("错误:传入的JSON字符串为空\n");return;}root = cJSON_Parse(message);if (!root) {printf("JSON格式错误:%s\n", cJSON_GetErrorPtr());return;}// 遍历JSON对象中的键值对(修正变量声明位置)item = root->child; // 变量声明已在代码块开头,此处直接使用while (item) {const char *device_name = item->string;const char *state = item->valuestring;// 查找设备处理器DeviceHandler *handler = device_handlers;while (handler->device_name) {if (strcmp(device_name, handler->device_name) == 0) {handler->handler(state);break;}handler++;}if (!handler->device_name) { // 未找到处理器printf("Unknown device: %s\r\n", device_name);}item = item->next;}cJSON_Delete(root); // 释放内存
}int main() {// 初始化部分保持不变...SysTick_Init(72);NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);USART1_Init(9600);LED_Init(); // 假设LED初始化函数在led.c中// Motor_Init(); // 如有电机初始化需添加WiFi_Config();printf("Hello, world!\r\n");printf("\r\n====================================\r\n");printf("STM32 + ESP8266 JSON Command Handler\r\n");printf("Supported devices: Led, Motor\r\n");printf("====================================\r\n");// 初始化WiFi和MQTTESP8266_STA_TCP_Client_MQTT();if (ESP8266_Set_MQTT_Sub(sub_topic_id, "0")) {printf("\r\nSubscribed to MQTT topic successfully!\r\n");} else {printf("\r\nFailed to subscribe to MQTT topic!\r\n");}if (ESP8266_MQTT_Public_key_Value(topic_id, "aa", "01")) {printf("\r\nPublished MQTT message successfully!\r\n");} else {printf("\r\nFailed to publish MQTT message!\r\n");}while (1) {// 处理WiFi数据if (strEsp8266_Fram_Record.InfBit.FramFinishFlag) {uint16_t data_len; // 声明在代码块开头__disable_irq();   // 进入临界区data_len = strEsp8266_Fram_Record.InfBit.FramLength;// 复制数据到本地缓冲区memcpy(wifi_rx_buffer, strEsp8266_Fram_Record.Data_RX_BUF, data_len);wifi_rx_buffer[data_len] = '\0'; // 添加字符串结束符// 清除标志和计数器strEsp8266_Fram_Record.InfBit.FramFinishFlag = 0;strEsp8266_Fram_Record.InfBit.FramLength = 0;__enable_irq(); // 退出临界区// 处理接收到的数据ProcessWiFiData(wifi_rx_buffer, data_len);}// 其他业务逻辑...delay_ms(100);}// 添加换行符避免警告printf("\n");
}// 处理WiFi数据的函数
void ProcessWiFiData(uint8_t *data, uint16_t length) {// 示例:直接打印原始数据printf("[RAW DATA] %s\r\n", data);// 检查是否是MQTT消息if (strstr((char *)data, "+MQTTSUBRECV:") != NULL) {// 解析MQTT消息char *topic_start = strchr((char *)data, ',') + 1;char *msg_length_start = strchr(topic_start, ',') + 1;char *msg_start = strchr(msg_length_start, ',') + 1;// 提取主题和消息char topic[32] = {0};char msg_length[32] = {0};char message[128] = {0};sscanf(topic_start, "%[^,]", topic);sscanf(msg_length_start, "%[^,]", msg_length);sscanf(msg_start, "%[^,]", message);HandleMQTTMessage(topic, message, msg_length);}// 可以添加其他协议处理...
}// 处理MQTT消息
void HandleMQTTMessage(const char *topic, const char *message, const char *message_length) {printf("Received MQTT message:\r\n");printf("Topic: %s\r\n", topic);         // topic_idprintf("Message: %s\r\n\r\n", message); // json 字符串// todo: 根据实际需求处理消息parse_json_message(message); // 解析JSON消息
}
// 添加换行符避免警告

用cjson 解析:

参考 STM32单片机与cJSON:构建并解析JSON数据_stm32 cjson-CSDN博客

// ========== JSON解析函数扩展 ==========
void parse_json_message(const char *message) {cJSON *root;cJSON *item; // 修正变量名,避免未使用警告if (message == NULL) {printf("错误:传入的JSON字符串为空\n");return;}root = cJSON_Parse(message);if (!root) {printf("JSON格式错误:%s\n", cJSON_GetErrorPtr());return;}// 遍历JSON对象中的键值对(修正变量声明位置)item = root->child; // 变量声明已在代码块开头,此处直接使用while (item) {const char *device_name = item->string;const char *state = item->valuestring;// 查找设备处理器DeviceHandler *handler = device_handlers;while (handler->device_name) {if (strcmp(device_name, handler->device_name) == 0) {handler->handler(state);break;}handler++;}if (!handler->device_name) { // 未找到处理器printf("Unknown device: %s\r\n", device_name);}item = item->next;}cJSON_Delete(root); // 释放内存
}

传递 

http://www.lryc.cn/news/2394750.html

相关文章:

  • 随笔笔记记录5.28
  • 大数据-273 Spark MLib - 基础介绍 机器学习算法 决策树 分类原则 分类原理 基尼系数 熵
  • 基于 Spring Boot + Vue 的墙绘产品展示交易平台设计与实现【含源码+文档】
  • 【机器学习】支持向量机
  • ONLYOFFICE深度解锁系列.4-OnlyOffice客户端原理-真的不支持多端同步
  • LLMTIME: 不用微调!如何用大模型玩转时间序列预测?
  • 2.从0开始搭建vue项目(node.js,vue3,Ts,ES6)
  • MySQL 高可用实现方案详解
  • 【pycharm】如何连接远程仓库进行版本管理(应用版本)
  • linux 1.0.7
  • 【Rust 轻松构建轻量级多端桌面应用】
  • IEEE P370:用于高达 50 GHz 互连的夹具设计和数据质量公制标准
  • 青少年编程与数学 02-020 C#程序设计基础 09课题、面向对象编程
  • Denoising Autoencoders 视频截图 DAEs简单实现 kaggle 去噪编码器
  • GoogLeNet网络模型
  • LeetCode Hot100 (贪心)
  • 仿真科普|弥合市场需求断层,高性能仿真,“性能”与“安全”如何兼得?
  • 工业控制核心引擎高性能MCU——MM32F5370
  • Maven---配置本地仓库
  • vue中events选项与$on监听自定义事件他们的区别与不同,以及$emit与$on之间通信和mounted生命周期钩子函数有哪些作用和属性
  • 【C++ 】智能指针:内存管理的 “自动导航仪”
  • 设备制造行业项目管理难点解析,如何有效解决?
  • 浅谈 PAM-2 到 PAM-4 的信令技术演变
  • Protos-SIP:经典 SIP 协议模糊测试工具!全参数详细教程!Kali Linux教程!
  • 复数三角不等式简介及 MATLAB 演示
  • 【Doris基础】Apache Doris 基本架构深度解析:从存储到查询的完整技术演进
  • 程序人生-hellohelloo
  • ASP.NET Core SignalR的基本使用
  • 【C语言】讲解 程序分配的区域(新手)
  • 【脚本 完全参数化的通用 APT 源配置方案-Debian/Ubuntu】