通过mqtt 点灯

article/2025/6/19 16:28:42

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.hkcw.cn/article/WUSotTYXxo.shtml

相关文章

解锁技术世界的“秘密知识库”:The Book of Secret Knowledge 深度解析

在浩如烟海的技术文档中,你是否渴望一个集中式宝库,收录那些资深工程师口耳相传的“秘密武器”?GitHub 上爆火的 The Book of Secret Knowledge 正是这样一个令人惊叹的集合。今天我们来深入探索这个项目,挖掘它的核心价值。 🔍 项目核心:不是什么,而是什么 不是一本传…

M4Pro安装ELK(ElasticSearch+LogStash+Kibana)踩坑记录

ElasticSearch安装&#xff0c;启动端口9200&#xff1a; docker pull elasticsearch:8.13.0 新增配置文件elasticsearch.yml&#xff1a; cd /opt/homebrew/etc/ mkdir elasticsearch_config cd elasticsearch_config vi elasticsearch.yml cluster.name: "nfturbo…

VC++: identifer “M_PI“ is undefined

本周拿到一份算法文件&#xff08;cpp),尝试在本机跑一下&#xff0c;提示M_PI不识别&#xff1a; identifer "M_PI" is undefined 解决方案&#xff1a; #define _USE_MATH_DEFINES

关于镜像如何装进虚拟机

本篇文章为感谢小仙猪老师特别编写 本篇文章仅以Ubuntu为例 目录 创建虚拟机 汉化 如果没有China选项 检查网络 创建虚拟机 第一步&#xff0c;创建虚拟机 因为&#xff0c;第一个选项是会把虚拟机的文件放在c盘因此&#xff0c;这里博主选择自定义&#xff0c;然后下一…

青柠日记:记录美好,守护隐私

在快节奏的现代生活中&#xff0c;日记成为了许多人记录生活、抒发情感的重要方式。而青柠日记这款软件&#xff0c;以其便捷的操作、丰富的模板和强大的隐私保护功能&#xff0c;为用户提供了一个理想的日记记录平台。无论是日常的琐碎点滴&#xff0c;还是重要的生活事件&…

【技能拾遗】——家庭宽带单线复用布线与配置(移动2025版)

&#x1f4d6; 前言&#xff1a;在家庭网络拓扑中&#xff0c;客厅到弱电箱只预埋了一根网线&#xff0c;由于已将广电的有线电视取消并改用IPTV。现在需要解决在客厅布置路由器和观看IPTV问题&#xff0c;这里就用到单线复用技术。 目录 &#x1f552; 1. 拓扑规划&#x1f55…

Visual Studio笔记:MSVC工具集、MSBuild

1. MSVC工具集 1.1 什么叫MSVC工具集 也可以说Visual Studio平台工具集&#xff08;Platform toolset&#xff09;. 这些工具包括 C/C 编译器、链接器、汇编程序和其他生成工具以及匹配的库和头文件。 Visual Studio 2015、Visual Studio 2017 和 Visual Studio 2019 是二进制…

【系统配置与部署类】docker的深度配置和应用

相关文章已经在个人博客网站上更新&#xff0c;欢迎访问&#xff1a; docker的深度配置和应用http://www.turnin-blog.online/articles/%E7%B3%BB%E7%BB%9F%E9%85%8D%E7%BD%AE%E4%B8%8E%E9%83%A8%E7%BD%B2/docker%E7%9A%84%E6%B7%B1%E5%BA%A6%E9%85%8D%E7%BD%AE%E5%92%8C%E5%B…

Redis最佳实践——安全与稳定性保障之数据持久化详解

Redis 在电商应用的安全与稳定性保障之数据持久化全面详解 一、持久化机制深度解析 1. 持久化策略矩阵 策略触发方式数据完整性恢复速度适用场景RDB定时快照分钟级快容灾备份/快速恢复AOF实时追加日志秒级慢金融交易/订单关键操作混合模式RDBAOF同时启用秒级中等高安全要求场…

告别硬编码!用工厂模式优雅构建可扩展的 Spring Boot 应用 [特殊字符]

嗨&#xff0c;各位技术伙伴们&#xff01;&#x1f44b; 在日常的软件开发中&#xff0c;我们经常面临需求变更的挑战。如何构建一个既能满足当前需求&#xff0c;又能轻松应对未来变化的系统呢&#xff1f;答案往往藏在那些经典的设计模式中。 今天&#xff0c;我们就来聊聊…

azure web app创建分步指南系列之二

为注册表授权托管标识 你创建的托管标识尚未获得从容器注册表中提取数据的授权。在此步骤中,你将启用授权。 返回容器注册表的管理页面: 在左侧导航菜单中,选择“访问控制 (IAM)”。选择“添加角色分配”。此屏幕截图显示了如何为容器注册表启用添加角色分配。在角色列表中…

使用Yolov8 训练交通标志数据集:TT100K数据集划分

使用Yolov8 训练交通标志数据集&#xff1a;TT100K数据集划分&#xff08;一&#xff09; 一、数据集下载二、划分数据集三、目录放置 一、数据集下载 官方网址&#xff1a;TT100K 数据集对比 源码如下&#xff1a; def classes(filedir):with open(filedir) as f:classes …

【PostgreSQL 03】PostGIS空间数据深度实战:从地图服务到智慧城市

PostGIS空间数据深度实战&#xff1a;从地图服务到智慧城市 关键词 PostGIS, 空间数据库, 地理信息系统, GIS, 空间查询, 地理分析, 位置服务, 智慧城市, 空间索引, 坐标系统 摘要 PostGIS是PostgreSQL的空间数据扩展&#xff0c;它将普通的关系数据库转变为强大的地理信息系统…

Wireshark 使用教程:让抓包不再神秘

一、什么是 tshark&#xff1f; tshark 是 Wireshark 的命令行版本&#xff0c;支持几乎所有 Wireshark 的核心功能。它可以用来&#xff1a; 抓包并保存为 pcap 文件 实时显示数据包信息 提取指定字段进行分析 配合 shell 脚本完成自动化任务 二、安装与验证 Kali Linux…

环境变量Path单行显示改回多行列表显示

环境变量Path单行显示改回多行列表显示 今天去配置环境变量时&#xff0c;双击Path竟然只显示一行&#xff0c;明明记得上次还时一个列表显示来着。由于以前有删除了所有Path变量的经历&#xff0c;所以看到这个情况属实吓我一跳且一脸懵。 仔细地看了一下&#xff0c;Path中…

CodeTop100 Day18

52、最长的有效括号 括号问题需要考虑栈&#xff0c;而最长有效的括号就要考虑动态规划了 这里定义dp[i]为以i-1位置结尾的最长合法括号字串 遍历字符串&#xff0c;当遇到左括号&#xff0c;压入栈&#xff0c;dp[i1]0,遇到右括号&#xff0c;如果栈不为空&#xff0c;配对左…

NLP基础:从词嵌入到预训练模型应用

NLP基础&#xff1a;从词嵌入到预训练模型应用 系统化学习人工智能网站&#xff08;收藏&#xff09;&#xff1a;https://www.captainbed.cn/flu 文章目录 NLP基础&#xff1a;从词嵌入到预训练模型应用摘要引言一、词嵌入技术&#xff1a;从离散到连续的语义表示1. 传统词嵌…

STM32CubeMX串口配置

STM32CubeMX串口配置 一&#xff0c;Mode1&#xff0c;Asynchronous&#xff08;异步模式&#xff09;2&#xff0c;其他模式3&#xff0c;异步通信中的流控 二&#xff0c;Configuration参数配置(Parameter Settings)1&#xff0c;Basic Parameters2&#xff0c;Advanced Para…

Java内存模型(JMM)与多线程编程实战

最近正在复习Java八股&#xff0c;所以会将一些热门的八股问题&#xff0c;结合ai与自身理解写成博客便于记忆 今天将以这四个问题为依据。 一、JMM内存模型解析 1. JMM核心概念 Java内存模型(Java Memory Model)定义了Java程序中各种变量&#xff08;线程共享变量&#xf…

Spring Cache核心原理与快速入门指南

文章目录 前言一、Spring Cache核心原理1.1 架构设计思想1.2 运行时执行流程1.3 核心组件协作1.4 关键机制详解1.5 扩展点设计1.6 与Spring事务的协同 二、快速入门实战三、局限性3.1 多级缓存一致性缺陷3.2 分布式锁能力缺失3.3 事务集成陷阱 总结 前言 在当今高并发、低延迟…