• 轻量规则引擎:让边缘设备自主决策
    轻量规则引擎:让边缘设备自主决策别再把所有“如果…就…”逻辑都扔给云端。我们在资源受限的 MCU(如 STM32F4、ESP32)上实现了一个轻量级规则引擎,支持本地实时联动——无需脚本、不依赖堆内存、启动即用。为什么需要它?断网不失效:网络中断时仍能执行安全策略(如超温关机)省流量:原始数据不出设备,只上报事件(如“高温告警”)低延迟:GPIO 响应 <100μs,远快于云端回路(>100ms)目标平台:RAM ≤64KB,主频 ≥80MHz,可运行 FreeRTOS 或裸机。核心设计:极简三原则1. 规则静态化,拒绝动态脚本不用 Lua/JS,避免解释器开销。规则以 C 结构体数组形式编译进固件:typedef struct { const char* point; // 数据点名,如 "temperature" float threshold; // 阈值 uint8_t op; // 操作符:OP_GT, OP_LT, OP_EQ uint8_t action; // 动作类型 uint8_t target; // 目标(如 GPIO 编号) } rule_t; static const rule_t rules[] = { {"temperature", 30.0f, OP_GT, ACTION_GPIO_HIGH, 5} }; 规则在编译期确定,运行时只读,杜绝注入风险。2. 解析即遍历,无 AST引擎核心是一个线性状态机:void evaluate_rules(const char* point, float value) { for (int i = 0; i < RULE_COUNT; i++) { if (strcmp(rules[i].point, point) == 0) { bool cond = false; switch (rules[i].op) { case OP_GT: cond = (value > rules[i].threshold); break; case OP_LT: cond = (value < rules[i].threshold); break; // ... } if (cond) execute_action(rules[i].action, rules[i].target); } } }  时间复杂度 O(n),n=规则数。实测 20 条规则评估耗时 <1ms。3. 零动态内存分配规则表存于 .rodata 段输入数据来自通信缓冲区(如 MQTT payload)所有变量使用栈或全局静态区启动后无 malloc,内存占用恒定。支持能力边界功能支持情况比较运算>, <, ==(浮点/整型)✅多条件通过多条规则组合模拟 AND ✅防抖“连续 N 次超限才触发”,每规则 +4 字节计数器 ✅动作类型GPIO 控制、发送 CoAP/MQTT、启定时器 ✅数学表达式❌(需云端预处理)循环/函数❌原则:边缘只做布尔判断,复杂逻辑上云。实测性能(ESP32 @ 240MHz)引擎代码体积:≈7KB运行时 RAM 占用:1.3KB(含 15 条规则)单规则评估:≈50μs启动初始化:几乎为零(静态数据)安全与部署动作白名单:仅开放 set_gpio()、publish_event() 等安全接口OTA 更新:新规则随固件下发,需 ECDSA 验签优先级控制:高优规则放数组前部,先匹配先执行典型场景工业终端:振动值突增 → 触发本地停机 + 上报智能插座:电流 >10A 持续 5 秒 → 自动断电农业网关:土壤湿度 <20% → 启动水泵 10 分钟结语这不是通用规则引擎,而是为边缘自治量身打造的有限智能模块。它用确定性和低开销,换取在资源牢笼中的可靠决策能力。对于大多数 IoT 场景,这已足够。
  • [技术干货] 边缘也能“思考”:在资源受限设备上实现轻量级规则引擎
    别以为“如果温度过高就关机”这种逻辑必须依赖云端。在 ESP32、STM32 这类只有几十 KB 内存的设备上,完全可以在本地实时执行联动逻辑——关键在于极简设计:无脚本、无堆分配、无 AST,只靠状态机和静态配置。为什么需要本地规则引擎?断网可用:网络中断时仍能响应紧急事件省流量:原始数据不出设备,只上报决策结果低延迟:GPIO 控制响应 <1ms,远快于云端回路目标很明确:在 ≤32KB RAM、≤100MHz 的 MCU 上稳定运行,启动开销可忽略。核心设计原则1. 规则即配置,禁止动态脚本放弃 JavaScript 或 Lua(太重)。改用结构化 JSON 定义规则:{ "id": "fan_control", "condition": { "point": "temperature", "operator": ">", "value": 30.0 }, "action": { "target": "gpio_5", "set": "high" } } 规则在固件编译时或通过安全 OTA 注入,运行时只读,杜绝注入风险。2. 解析器 = 线性状态机不构建抽象语法树(AST),而是遍历规则列表做字段匹配:收到遥测数据 {“temperature”: 32.1}遍历所有规则,检查 condition.point == “temperature”若匹配,执行比较(32.1 > 30.0)条件成立 → 执行预定义动作(如拉高 GPIO)全程无递归、无动态内存,时间复杂度 O(n),n 为规则数。3. 零堆内存,全静态布局规则数组放在 .rodata 段输入数据复用通信缓冲区(如 MQTT payload buffer)比较操作仅使用栈上局部变量实测(ESP32 + FreeRTOS):代码体积:≈8KB运行时 RAM:≈1.2KB(支持 10 条规则)单规则评估耗时:<50μs支持的能力边界功能 支持情况比较运算 >, <, ==, != ✅多条件 AND 组合 ✅动作类型 GPIO、发消息、启定时器 ✅防抖 “持续超限 X 秒才触发” ✅数学表达式 ❌(留给云端)自定义函数 ❌原则:边缘只做布尔判断,复杂逻辑上云。落地技巧热更新:通过签名验证的 MQTT 消息下发新规则,写入备份 Flash 区后切换优先级:按数组顺序执行,高优规则前置安全沙箱:动作接口白名单化,禁止直接调用系统 API典型场景智能断路器:电流突增 → 立即切断 + 上报告警温室控制器:湿度 < 30% → 开启加湿 10 分钟工业终端:设备停机 → 触发本地蜂鸣器 + 推送状态变更结语这不是“缩水版业务引擎”,而是为边缘重新定义的有限智能模块。它用确定性换灵活性,在资源牢笼中实现可靠自治。对于大多数 IoT 场景,这已足够。
  • TSN 如何赋能确定性工业物联网?
    TSN:让以太网真正扛起工业控制别再用“普通以太网+尽力而为”做实时控制了。工业场景要的是确定性——不是“大概几毫秒”,而是“必须在 200μs 内完成”。TSN(Time-Sensitive Networking)就是干这个的:把标准以太网改造成硬实时通信底座。为什么传统以太网不行?交换机缓存排队 → 延迟不可控长帧阻塞短帧 → 急停指令被卡住时钟不同步 → 多轴电机失配结果:抖动大、不可靠,只能靠专用总线(如 EtherCAT、Profinet)兜底。但专用协议成本高、生态封闭。TSN 的目标:用一根网线,同时跑控制流和 IT 流。TSN 三大核心机制,缺一不可1. 时间同步:802.1AS(gPTP)所有节点(包括交换机)对齐到同一个全局时钟精度 ±50ns,远超 NTP(ms 级)和普通 PTP同步是调度的前提——没有统一时间表,门控无从谈起2. 时间门控:802.1Qbv(TAS)在交换机出口队列前加“时间闸门”按微秒级时间表开关:[0–100μs]:只放行高优控制帧[100–900μs]:放行监控/配置数据[900–1000μs]:保护带(防抖动溢出)关键帧零排队,端到端延迟稳定在 200μs ±10μs3. 帧抢占:802.1Qbu + 802.3br高优先级短帧可打断低优先级长帧传输被打断的帧后续自动续传,MAC 层无感知解决“一个 1500 字节背景包卡住整个链路”的经典问题实测:启用抢占后,100 字节控制帧延迟从 120μs 降至 8μs。和 OPC UA 怎么配合?OPC UA 提供统一语义模型(谁发什么数据)TSN 提供确定性通道(什么时候必须送到)合体 = OPC UA PubSub over TSN优势:IT/OT 融合,同一网络承载 PLC 控制 + MES 数据 + 视频监控主流工业联盟(PI、ODVA、EPSG)已将其列为下一代标准。落地现状✅ 能用:Intel TCC、NXP S32G、TI Sitara 已支持硬件级 TSN⚠️ 难点:时间表需离线规划(工具链尚不成熟)老设备需 TSN 网关转换运维人员需新技能栈结论TSN 不是“又一个网络协议”,而是工业通信基础设施的范式转移。如果你在设计新一代产线、机器人或电力控制系统,现在就该评估 TSN 架构。未来 3–5 年,它将逐步替代传统现场总线。
  • CoAP over DTLS:低功耗设备的安全通信实现
    CoAP over DTLS:低功耗设备的安全通信实现在 NB-IoT、LoRaWAN 等 LPWAN 场景中,终端设备常运行于 Cortex-M 系列 MCU,内存低于 64KB,无法承受传统 TLS 的资源开销。此时,CoAP(Constrained Application Protocol)结合 DTLS(Datagram Transport Layer Security) 成为兼顾安全性与低功耗的最优解。为何选择 DTLS 而非 TLS?CoAP 基于 UDP,而标准 TLS 依赖 TCP 的可靠流传输。若强行在 CoAP 上叠加 TLS,需引入虚拟流层,极大增加复杂度与延迟。DTLS 是 TLS 的 UDP 友好版本,保留其加密、认证、完整性保护能力,同时适配无连接特性。关键差异:DTLS 在记录层添加 序列号 和 重传机制,防止 UDP 丢包导致握手失败。四大优化策略实现实用化部署1. 证书轻量化:ECC 替代 RSARSA 2048 位证书 ≈ 2KBECC(ECDSA-P256)证书 ≈ 300 字节在 STM32L4 上实测,ECC 验签耗时 <50ms,内存占用减少 60%。建议使用 secp256r1 曲线,兼容性最佳。2. PSK 模式:封闭系统的高效选择对于工厂内网、私有平台等可信环境,可采用 预共享密钥(Pre-Shared Key, PSK) 模式:无需证书交换,握手时间从 2s 缩短至 200ms密钥可烧录至设备安全存储区(如 STM32 OTP)注意:PSK 需配合唯一设备 ID 使用,避免密钥复用导致横向渗透。3. 会话复用:降低重复握手开销首次 DTLS 握手后,缓存 Session ID 或 Session Ticket。后续通信直接恢复会话,跳过密钥协商:首次握手延迟:~1.8s复用后延迟:<10ms适用于周期性上报场景(如每 5 分钟上传一次数据)。4. 安全块传输:防中间人篡改CoAP 支持 Block-wise Transfer 分块传输大消息。必须确保每个块独立加密,否则攻击者可替换中间块(如篡改固件片段)。mbed TLS 等库需启用 MBEDTLS_SSL_DTLS_CONNECTION_ID 以保障分片一致性。性能实测(ESP32 + mbedtls 3.4)模式平均 CPU 占用内存峰值单次通信延迟纯 CoAP8%12 KB15 msCoAP + DTLS (ECC)23%28 KB65 msCoAP + DTLS (PSK)18%22 KB35 ms尽管资源开销增加,但满足《网络安全等级保护 2.0》对物联网终端的加密通信要求。部署注意事项禁用弱加密套件:如 TLS_PSK_WITH_AES_128_CCM_8 可接受,但避免 NULL 或 RC4 算法。证书生命周期管理:通过 OTA 更新根证书,避免硬编码过期 CA。模组兼容性:部分 NB-IoT 模组(如 BC95)仅支持 DTLS 1.0,需确认安全协议版本。结语CoAP over DTLS 并非“开箱即用”,而是需要在安全强度、功耗、延迟、资源之间精细权衡。但在当前国产芯片普遍支持硬件加密引擎(如 AES、TRNG)的背景下,该方案已成为中高安全等级 LPWAN 设备的事实标准。应用层:CoAP(GET/POST)安全层:DTLS(Handshake / Record)传输层:UDP网络层:IPv6/6LoWPAN在各层旁标注典型资源消耗(如 DTLS 层:+16KB RAM, +15% CPU)。
  • [专题汇总] 2025年11月嵌入式项目开发专题总汇
    *一、前言*我们正处在一个由嵌入式智能与泛在连接所驱动的技术革新时代。微控制器、传感器、无线通信和人工智能技术的深度融合,正以前所未有的力量将数字世界的智慧注入物理世界的各个角落,催生出无数提升效率、保障安全、改善生活的创新解决方案。本书收录的技术文章,正是这一趋势的生动缩影与实践典范。本系列项目围绕****STM32**** 等主流嵌入式平台展开,充分展现了其作为“智慧大脑”在复杂场景下的核心控制能力。同时,项目广泛集成了****4G、Zigbee、北斗*等多元通信技术,突破了信息传输的空间限制,构建了从本地到远程的立体化感知与控制网络。更为亮眼的是,*人工智能(AI)**** 的引入为传统嵌入式设备装上了“智慧之眼”与“思考之芯”。从智能导盲杖的环境感知,到垃圾分类的视觉识别,AI算法让设备具备了理解与决策的能力,将自动化提升至智能化的新高度。这些技术并非空中楼阁,它们精准地服务于各行各业的关键需求:· 在****工业与能源领域****,光伏板智能清洁与健康监测系统、智能仓储机器人致力于提升运营效率与能源效益;· 在****智慧生活领域****,智能家居、防久坐系统、智能垃圾桶等应用正悄然重塑我们的日常体验;· 在****健康与安全领域****,便携式心电监护仪、户外遇险预警装置、智能导盲杖彰显了技术对个体生命健康的深切关怀;· 在****农业与环保领域****,智能农业与环境监测系统则展示了科技赋能可持续发展的重要路径。通过这些详实的项目案例,我们不仅旨在剖析技术实现的细节,更希望勾勒出一幅“万物互联、智能共生”的未来蓝图。无论您是嵌入式开发者、物联网爱好者还是行业创新者,都能从中汲取灵感,共同探索智能硬件技术的无限可能。二、文章总汇技术文章_ 基于4G的无人机远程控制系统(https://bbs.huaweicloud.com/forum/thread-0282199005687952007-1-1.html)本项目设计了一套基于4G移动网络的无人机远程控制系统,成功突破了传统无线电遥控在距离上的限制。系统利用4G网络的高带宽、低延迟和广覆盖特性,将无人机的实时视频数据、飞行姿态及传感器信息传输至远程控制端,同时接收来自控制端的飞行指令,实现了在互联网可达范围内的超视距精准控制,为物流、巡检、测绘等工业级应用奠定了基础。技术文章_AI垃圾分类助手(https://bbs.huaweicloud.com/forum/thread-0235199005917115006-1-1.html)该项目结合计算机视觉技术与嵌入式硬件,开发了一款智能垃圾分类助手。系统以摄像头捕捉垃圾图像,在集成AI加速引擎的嵌入式平台上运行深度学习模型,对垃圾进行快速识别与分类(如可回收物、厨余垃圾、有害垃圾等),并通过灯光、语音或屏幕提示等方式引导用户正确投放,有效解决了垃圾分类的难题,助推环保事业的智能化。技术文章_光伏板智能除尘系统(https://bbs.huaweicloud.com/forum/thread-0282199005972502008-1-1.html)针对光伏电站因表面积尘导致发电效率下降的问题,本系统设计了一套自动化除尘方案。它通过灰尘传感器监测光伏板表面的洁净度,或根据预设的时间策略,自动控制清洁机构(如滚刷、喷淋装置)进行定时或按需除尘,减少了人工维护成本,保障了光伏电站的稳定高效运行。技术文章_光伏板智能清洁与健康监测系统(https://bbs.huaweicloud.com/forum/thread-02127199006001720011-1-1.html)在基础除尘功能之上,本项目进一步集成了健康监测功能。系统不仅能够自动清洁光伏板,还能通过IV曲线扫描、红外热成像等技术实时监测光伏板的工作电压、电流、温度等关键参数,诊断其潜在故障(如热斑、隐裂),实现从“被动清洁”到“主动运维”的升级,为电站的预防性维护提供数据支持。技术文章_基于STM32的便携式心电监护仪(https://bbs.huaweicloud.com/forum/thread-02126199006044258010-1-1.html)本项目以高性能STM32单片机为核心,设计了一款便携式心电信号采集与监护设备。它通过导联电极采集人体微弱的心电信号,经放大、滤波、数字化处理后,可在本地OLED屏幕上实时显示心电图,并具备心率计算、心律失常初步判断、数据存储与蓝牙上传等功能,为个人心血管健康监护及远程医疗提供了便捷的工具。技术文章_基于STM32的智能仓储盘点机器人(https://bbs.huaweicloud.com/forum/thread-0250199006068717004-1-1.html)该智能机器人旨在解决传统仓储盘点工作强度大、效率低的问题。机器人以STM32为控制中枢,集成RFID读写器、里程计和避障传感器,能够自主或遥控巡弋于货架间,自动识别并读取货物上的RFID标签信息,实现库存的快速、精准、自动化盘点,极大提升了仓储管理的智能化水平。技术文章_基于STM32的智能防久坐与健康办公系统(https://bbs.huaweicloud.com/forum/thread-0213199006093327008-1-1.html)关注办公室人群的健康,该系统通过STM32连接人体红外传感器、压力垫片等,检测用户是否长时间处于坐姿状态。当持续久坐超过设定时限,系统会通过灯光、声音或轻微震动等方式发出提醒,并可能联动升降桌鼓励用户站立活动,有效促进健康办公习惯的养成。8. 技术文章_基于STM32的智能垃圾桶(https://bbs.huaweicloud.com/forum/thread-0213199006147335009-1-1.html)本项目对传统垃圾桶进行了智能化改造。利用STM32控制红外传感器、超声波测距模块和舵机,实现了非接触式自动开盖、垃圾满溢报警等功能。部分高级版本还可集成称重模块,为量化个人垃圾产出提供数据,是智能家居场景中的一个实用创新。9.技术文章_基于STM32的智能农业环境监测系统(https://bbs.huaweicloud.com/forum/thread-02126199006173832011-1-1.html)该系统面向精准农业需求,通过STM32采集部署在田间地头的多种传感器数据,包括空气温湿度、土壤湿度、光照强度、二氧化碳浓度等。数据可通过LoRa、NB-IoT等无线方式上传至云平台,帮助农民实时掌握作物生长环境,为实现智能灌溉、科学施肥提供决策依据。10. 技术文章_基于STM32的智能鱼塘监控系统(https://bbs.huaweicloud.com/forum/thread-02117199006201398008-1-1.html)专注于水产养殖,本系统通过STM32实时监控鱼塘水体的关键参数,如溶解氧、pH值、水温等。一旦检测到数值异常(如缺氧),系统可自动启动增氧机等设备,并能远程向养殖户报警,有效规避养殖风险,保障水产品的存活与品质。技术文章_基于STM32与AI的导盲杖系统(https://bbs.huaweicloud.com/forum/thread-0235199006225566007-1-1.html)这是一款增强传统导盲杖功能的辅助设备。在STM32主控上,集成了超声波、摄像头乃至激光雷达等传感器,利用轻量级AI算法识别前方的障碍物、坑洼、台阶等,并通过语音或振动反馈给视障人士,极大地提升了出行的安全性与独立性。技术文章_基于Zigbee的智能家居灯光控制系统(https://bbs.huaweicloud.com/forum/thread-0293199006254085012-1-1.html)本项目构建了一个基于Zigbee无线组网技术的智能灯光控制系统。系统由Zigbee协调器、智能开关/调光器以及移动终端APP组成,实现了对家庭灯光的无线开关、亮度调节、情景模式切换及远程控制,具备低功耗、自组网、高可靠性等特点,是智能家居的典型应用。技术文章_基于北斗的户外遇险预警定位装置https://bbs.huaweicloud.com/forum/thread-0293199006326044013-1-1.html)针对登山、探险等户外活动中的安全问题,本装置以北斗卫星导航系统为核心,能够实时获取并回传使用者的精准位置信息。在遇险时,用户可一键触发SOS警报,将包含精确经纬度的求救信号发送至救援中心,大大缩短了救援队伍的搜寻时间,为生命安全增添了一份保障。技术文章_基于单片机的CAN总线车辆信息监测仪(https://bbs.huaweicloud.com/forum/thread-0250199006359864005-1-1.html)本项目开发了一款用于车载网络的诊断与监测设备。它通过单片机连接汽车的CAN总线,能够读取并解析发动机转速、车速、冷却液温度、故障码等丰富的车辆运行参数,并在显示屏上直观展示,为车辆状态监控、故障诊断及车联网数据应用提供了硬件基础。技术文章_基于单片机的无线红外遥控密码锁(https://bbs.huaweicloud.com/forum/thread-0235199006393591009-1-1.html)该设计将传统电子密码锁与无线技术结合。用户既可通过本地键盘输入密码开锁,也能利用红外遥控器在一定距离内实现无线开锁,增强了使用的便利性。系统具备密码错误报警、管理密码设置等功能,在安全与便捷之间取得了良好平衡。技术文章_实验室紫外灯智能消毒机器人(https://bbs.huaweicloud.com/forum/thread-02126199006423029012-1-1.html)为实现特定区域(如实验室、病房)的自动化消毒,本机器人集成了自主移动底盘与紫外灯管。它可采用预设路径或遥控方式进入目标区域,在确保无人的情况下自动开启紫外灯进行消杀,任务完成后自动返回充电,有效避免了人员接触紫外线的风险,提升了消毒工作的效率和安全性。17. 技术文章_智能自行车状态监测与防盗系统(https://bbs.huaweicloud.com/forum/thread-02127199006452018012-1-1.html)该系统为自行车增加了智能化维度。通过STM32连接GPS定位模块、GSM通信模块和振动传感器,可实时监测车辆的位置、行驶轨迹和状态。当发生异常移动(盗窃)时,系统会立即向车主手机发送警报和位置信息,并具备电子围栏功能,实现了有效的状态监测与防盗保护。
  • [技术干货] 智能自行车状态监测与防盗系统
    项目开发背景随着城市化进程的加速和环保意识的提升,自行车作为一种绿色、便捷的出行工具,在全球范围内受到广泛欢迎。然而,自行车盗窃问题日益严重,给用户带来财产损失和不便,传统机械锁具的防护能力有限,难以应对复杂的盗窃手段。此外,用户对骑行数据记录和实时状态监控的需求也在增长,这促使了智能自行车系统的开发。在现有市场中,自行车防盗和状态监测方案多依赖于简单的物理锁具或独立的电子设备,缺乏集成化和智能化。这些方案无法实时感知车辆异常移动、追踪位置或提供全面的骑行数据分析,导致用户无法及时响应盗窃事件或优化骑行体验。因此,亟需一种融合多传感器和通信技术的综合解决方案。本项目基于现代物联网和嵌入式系统技术,旨在设计一款智能自行车状态监测与防盗系统。通过集成霍尔传感器、MPU6050姿态传感器和GPS模块,系统能够实时计算速度与里程、监测车身姿态并记录骑行轨迹。同时,利用Wi-Fi/蓝牙模块和电磁锁实现远程警报和锁定功能,有效提升防盗能力,并通过手机APP和本地显示屏提供直观的数据交互。该系统的开发不仅响应了用户对自行车安全和便捷管理的需求,还推动了智能交通设备的发展。通过结合低成本硬件和高效算法,项目有望降低自行车盗窃率,增强骑行体验,并为智慧城市出行生态的构建提供支持。设计实现的功能(1)通过霍尔传感器计算车轮转速,进而得到实时速度与里程。(2)通过MPU6050监测车身姿态,当判定为异常移动(被盗)时,锁定电子锁并向车主手机发送警报。(3)通过GPS模块记录骑行轨迹,并可通过手机APP查看。(4)所有数据可在本地OLED屏幕上显示。项目硬件模块组成(1) 主控芯片:STM32F103C8T6单片机。(2) 传感与定位:A3144霍尔传感器、MPU6050姿态传感器、GPS模块。(3) 通信与报警:ESP8266 Wi-Fi/蓝牙模块(或4G模块)、有源蜂鸣器。(4) 执行机构:继电器控制的电磁锁。设计意义该智能自行车状态监测与防盗系统的设计意义在于提升自行车的安全性和使用便利性。通过集成霍尔传感器和MPU6050姿态传感器,系统能够实时监测车轮转速和车身姿态,有效识别异常移动情况,从而及时锁定电子锁并发送警报,大大降低自行车被盗的风险,保护用户的财产安全。系统通过GPS模块记录骑行轨迹,并结合手机APP查看功能,使用户能够回顾和分析骑行路线,这不仅有助于日常通勤或休闲骑行的规划,还能为健身活动提供数据支持,增强骑行的趣味性和实用性。本地OLED屏幕显示所有关键数据,如速度和里程,使用户无需依赖外部设备即可快速获取信息,提高了系统的易用性和响应速度,同时减少了对外部网络的依赖,确保在无网络环境下仍能正常运作。整体设计整合了现代传感、通信和控制技术,体现了物联网在智能交通领域的应用潜力,为自行车智能化提供了可行方案,推动传统交通工具向更安全、高效的方向发展。设计思路本系统以STM32F103C8T6单片机作为核心控制器,负责协调各个硬件模块的工作,实现自行车状态监测与防盗功能。系统通过集成传感器、通信模块和执行机构,构建一个完整的智能监测网络。在速度与里程计算方面,使用A3144霍尔传感器检测车轮旋转,通过计算单位时间内脉冲次数来推导车轮转速,进而结合车轮周长计算出实时速度和累计里程。STM32主控芯片处理这些数据,确保计算准确性和实时性。对于防盗功能,MPU6050姿态传感器持续监测车身姿态变化,当检测到异常移动(如非正常倾斜或震动)时,STM32会判断为被盗状态,并触发继电器控制的电磁锁进行锁定,同时通过ESP8266 Wi-Fi/蓝牙模块向车主手机发送警报信息,配合有源蜂鸣器发出声音警示。GPS模块用于记录骑行轨迹,实时获取位置数据并存储在系统中,车主可通过手机APP查看历史轨迹。通信模块负责将GPS数据上传到云端或直接传输到手机,确保轨迹信息的可访问性和持久性。所有监测数据,包括速度、里程、姿态状态和位置信息,都会在本地OLED屏幕上实时显示,方便用户直观查看自行车当前状态。STM32主控统一管理数据采集、处理和显示流程,确保系统稳定运行。整个设计注重实际应用,硬件模块之间通过STM32进行高效协同,传感器数据采集、通信传输和执行控制各环节紧密衔接,形成一个可靠的状态监测与防盗系统。框架图+-------------------+ +-------------------+ | 霍尔传感器 |------>| | | (A3144) | | | +-------------------+ | | | 主控芯片 | +-------------------+ | (STM32F103C8T6) | | MPU6050 |------>| | | 姿态传感器 | | | +-------------------+ | | | | +-------------------+ | | | GPS模块 |------>| | | | +-------------------+ +-------------------+ | | v +-------------------+ +-------------------+ | OLED显示屏 |<------| | | | | 通信模块 | +-------------------+ | (ESP8266) | | | +-------------------+ | | | 蜂鸣器 |<------| | | (报警) | +-------------------+ +-------------------+ | | v +-------------------+ +-------------------+ | 电磁锁 |<------| 继电器 | | (通过继电器控制) | | | +-------------------+ +-------------------+ | v +-------------------+ | 手机APP | | (查看数据/警报) | +-------------------+ 系统总体设计智能自行车状态监测与防盗系统以STM32F103C8T6单片机作为核心控制器,负责协调和处理各个模块的数据与指令。系统通过集成多种传感器实现状态监测,其中A3144霍尔传感器用于检测车轮转动,通过计算脉冲频率得到实时速度与累计里程;MPU6050姿态传感器持续监测车身倾斜和移动状态,当检测到异常振动或位移时,系统判定为潜在盗窃行为,立即触发防盗机制。在防盗功能中,系统通过继电器控制电磁锁进行锁定,防止自行车被移动,同时利用ESP8266 Wi-Fi/蓝牙模块或4G模块将警报信息发送至车主手机,确保及时通知。此外,GPS模块实时记录骑行轨迹,数据通过通信模块上传至云端,车主可通过手机APP查看历史路径和当前位置。所有采集的数据,包括速度、里程、姿态信息和GPS位置,均可在本地OLED屏幕上实时显示,方便用户直观查看自行车状态。系统设计注重实际应用,各模块协同工作,实现了高效的监测、防盗和数据分析功能。系统功能总结功能关键硬件模块功能描述实时速度与里程计算A3144霍尔传感器、STM32F103C8T6通过霍尔传感器检测车轮转速,由主控芯片计算并输出实时速度与总里程。车身姿态监测与防盗警报MPU6050、继电器控制电磁锁、ESP8266、有源蜂鸣器监测车身姿态,当判定异常移动(如被盗)时,锁定电磁锁,并通过通信模块向手机发送警报,蜂鸣器发声提醒。骑行轨迹记录与查看GPS模块、ESP8266使用GPS模块记录骑行路径,数据通过通信模块传输,用户可通过手机APP查看历史轨迹。数据本地显示OLED屏幕(假设)、STM32F103C8T6所有监测数据(如速度、里程、姿态状态)在本地OLED屏幕上实时显示。设计的各个功能模块描述霍尔传感器模块用于监测自行车车轮的转速,通过A3144霍尔传感器检测车轮上安装的磁铁信号,主控芯片STM32F103C8T6处理这些脉冲信号,计算车轮转动频率,从而得出实时速度和累计里程数据。姿态监测与防盗模块利用MPU6050传感器检测车身的倾斜和加速度变化,当系统判断为异常移动时,主控芯片STM32F103C8T6会控制继电器驱动的电磁锁进行锁定,防止车辆被盗,同时通过ESP8266 Wi-Fi模块向车主手机发送警报通知,并触发有源蜂鸣器发出声音警示。GPS轨迹记录模块通过GPS模块获取自行车的实时位置信息,主控芯片STM32F103C8T6存储和处理这些数据,形成骑行轨迹记录,用户可以通过手机APP经由ESP8266模块查看历史路线和当前位置。数据显示模块将所有采集的数据,包括速度、里程、位置和系统状态,实时显示在本地连接的OLED屏幕上,为用户提供直观的信息反馈。上位机代码设计#include <iostream> #include <string> #include <vector> #include <thread> #include <chrono> #include <mutex> #include <map> #include <iomanip> #include <sstream> #include <winsock2.h> #include <windows.h> #pragma comment(lib, "ws2_32.lib") // 数据结构定义 struct BikeStatus { double speed; // 实时速度 km/h double distance; // 累计里程 km double roll; // 横滚角 double pitch; // 俯仰角 double latitude; // 纬度 double longitude; // 经度 bool isLocked; // 电子锁状态 bool alertTriggered; // 报警触发状态 std::string timestamp; // 时间戳 }; class BikeMonitor { private: SOCKET clientSocket; std::mutex dataMutex; BikeStatus currentStatus; bool isConnected; bool running; std::thread receiveThread; std::vector<std::pair<double, double>> gpsTrajectory; // GPS轨迹记录 public: BikeMonitor() : clientSocket(INVALID_SOCKET), isConnected(false), running(false) { initializeStatus(); } ~BikeMonitor() { disconnect(); } // 初始化状态数据 void initializeStatus() { currentStatus.speed = 0.0; currentStatus.distance = 0.0; currentStatus.roll = 0.0; currentStatus.pitch = 0.0; currentStatus.latitude = 0.0; currentStatus.longitude = 0.0; currentStatus.isLocked = false; currentStatus.alertTriggered = false; currentStatus.timestamp = getCurrentTime(); } // 连接到下位机 bool connectToDevice(const std::string& ip, int port) { WSADATA wsaData; if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) { std::cerr << "WSAStartup failed!" << std::endl; return false; } clientSocket = socket(AF_INET, SOCK_STREAM, 0); if (clientSocket == INVALID_SOCKET) { std::cerr << "Socket creation failed!" << std::endl; return false; } sockaddr_in serverAddr; serverAddr.sin_family = AF_INET; serverAddr.sin_port = htons(port); serverAddr.sin_addr.s_addr = inet_addr(ip.c_str()); if (connect(clientSocket, (sockaddr*)&serverAddr, sizeof(serverAddr)) == SOCKET_ERROR) { std::cerr << "Connection failed!" << std::endl; closesocket(clientSocket); return false; } isConnected = true; running = true; // 启动数据接收线程 receiveThread = std::thread(&BikeMonitor::receiveData, this); std::cout << "Connected to bike device successfully!" << std::endl; return true; } // 断开连接 void disconnect() { running = false; isConnected = false; if (receiveThread.joinable()) { receiveThread.join(); } if (clientSocket != INVALID_SOCKET) { closesocket(clientSocket); clientSocket = INVALID_SOCKET; } WSACleanup(); std::cout << "Disconnected from bike device." << std::endl; } // 发送控制命令 bool sendCommand(const std::string& command) { if (!isConnected) { std::cerr << "Not connected to device!" << std::endl; return false; } std::string fullCommand = command + "\n"; if (send(clientSocket, fullCommand.c_str(), fullCommand.length(), 0) == SOCKET_ERROR) { std::cerr << "Send command failed!" << std::endl; return false; } std::cout << "Command sent: " << command << std::endl; return true; } // 锁定电子锁 bool lockBike() { return sendCommand("LOCK"); } // 解锁电子锁 bool unlockBike() { return sendCommand("UNLOCK"); } // 获取当前状态 BikeStatus getCurrentStatus() { std::lock_guard<std::mutex> lock(dataMutex); return currentStatus; } // 获取GPS轨迹 std::vector<std::pair<double, double>> getGPSTrajectory() { std::lock_guard<std::mutex> lock(dataMutex); return gpsTrajectory; } // 显示状态信息 void displayStatus() { std::lock_guard<std::mutex> lock(dataMutex); std::cout << "\n=== 智能自行车状态监测 ===" << std::endl; std::cout << "时间: " << currentStatus.timestamp << std::endl; std::cout << "速度: " << std::fixed << std::setprecision(1) << currentStatus.speed << " km/h" << std::endl; std::cout << "里程: " << std::fixed << std::setprecision(2) << currentStatus.distance << " km" << std::endl; std::cout << "姿态 - 横滚: " << std::fixed << std::setprecision(2) << currentStatus.roll << "°, 俯仰: " << currentStatus.pitch << "°" << std::endl; std::cout << "GPS位置: " << std::fixed << std::setprecision(6) << currentStatus.latitude << ", " << currentStatus.longitude << std::endl; std::cout << "电子锁: " << (currentStatus.isLocked ? "锁定" : "解锁") << std::endl; std::cout << "报警状态: " << (currentStatus.alertTriggered ? "触发" : "正常") << std::endl; std::cout << "轨迹点数: " << gpsTrajectory.size() << std::endl; if (currentStatus.alertTriggered) { std::cout << "!!! 警报: 检测到异常移动 !!!" << std::endl; } } private: // 数据接收线程 void receiveData() { char buffer[1024]; int bytesReceived; while (running && isConnected) { bytesReceived = recv(clientSocket, buffer, sizeof(buffer) - 1, 0); if (bytesReceived > 0) { buffer[bytesReceived] = '\0'; processReceivedData(std::string(buffer)); } else if (bytesReceived == 0) { std::cout << "Connection closed by device." << std::endl; break; } else { std::cerr << "Receive data error!" << std::endl; break; } std::this_thread::sleep_for(std::chrono::milliseconds(100)); } } // 处理接收到的数据 void processReceivedData(const std::string& data) { std::lock_guard<std::mutex> lock(dataMutex); std::istringstream iss(data); std::string line; while (std::getline(iss, line)) { if (line.empty()) continue; // 解析数据格式: TYPE:VALUE1,VALUE2,... size_t colonPos = line.find(':'); if (colonPos == std::string::npos) continue; std::string dataType = line.substr(0, colonPos); std::string valuesStr = line.substr(colonPos + 1); std::istringstream valuesStream(valuesStr); std::string value; std::vector<std::string> valueList; while (std::getline(valuesStream, value, ',')) { valueList.push_back(value); } if (dataType == "SPEED" && valueList.size() >= 2) { currentStatus.speed = std::stod(valueList[0]); currentStatus.distance = std::stod(valueList[1]); } else if (dataType == "ATTITUDE" && valueList.size() >= 2) { currentStatus.roll = std::stod(valueList[0]); currentStatus.pitch = std::stod(valueList[1]); // 姿态异常检测 if (std::abs(currentStatus.roll) > 45 || std::abs(currentStatus.pitch) > 45) { currentStatus.alertTriggered = true; triggerAlarm(); } } else if (dataType == "GPS" && valueList.size() >= 2) { currentStatus.latitude = std::stod(valueList[0]); currentStatus.longitude = std::stod(valueList[1]); // 记录GPS轨迹 gpsTrajectory.push_back(std::make_pair( currentStatus.latitude, currentStatus.longitude)); } else if (dataType == "LOCK") { currentStatus.isLocked = true; } else if (dataType == "UNLOCK") { currentStatus.isLocked = false; } else if (dataType == "ALERT") { currentStatus.alertTriggered = true; triggerAlarm(); } currentStatus.timestamp = getCurrentTime(); } } // 触发警报 void triggerAlarm() { std::cout << "\n!!! 自行车防盗警报触发 !!!" << std::endl; std::cout << "检测到异常移动,请立即检查自行车状态!" << std::endl; std::cout << "位置: " << currentStatus.latitude << ", " << currentStatus.longitude << std::endl; // 在实际应用中,这里可以发送短信、邮件或推送通知 sendCommand("LOCK"); // 自动锁定 } // 获取当前时间字符串 std::string getCurrentTime() { auto now = std::chrono::system_clock::now(); auto time_t = std::chrono::system_clock::to_time_t(now); std::tm tm; localtime_s(&tm, &time_t); std::ostringstream oss; oss << std::put_time(&tm, "%Y-%m-%d %H:%M:%S"); return oss.str(); } }; // 主应用程序类 class BikeMonitorApp { private: BikeMonitor monitor; bool appRunning; public: BikeMonitorApp() : appRunning(false) {} void run() { appRunning = true; std::cout << "=== 智能自行车状态监测与防盗系统 ===" << std::endl; // 连接到设备 if (!connectToDevice()) { std::cerr << "Failed to connect to device. Exiting..." << std::endl; return; } // 主循环 mainLoop(); } void stop() { appRunning = false; monitor.disconnect(); } private: bool connectToDevice() { std::string ip; int port; std::cout << "请输入设备IP地址 (默认: 192.168.1.100): "; std::getline(std::cin, ip); if (ip.empty()) ip = "192.168.1.100"; std::cout << "请输入端口号 (默认: 8080): "; std::string portStr; std::getline(std::cin, portStr); port = portStr.empty() ? 8080 : std::stoi(portStr); return monitor.connectToDevice(ip, port); } void mainLoop() { while (appRunning) { displayMenu(); handleUserInput(); std::this_thread::sleep_for(std::chrono::milliseconds(500)); } } void displayMenu() { std::cout << "\n=== 主菜单 ===" << std::endl; std::cout << "1. 显示当前状态" << std::endl; std::cout << "2. 锁定自行车" << std::endl; std::cout << "3. 解锁自行车" << std::endl; std::cout << "4. 查看GPS轨迹" << std::endl; std::cout << "5. 清除警报" << std::endl; std::cout << "0. 退出程序" << std::endl; std::cout << "请选择操作: "; } void handleUserInput() { int choice; std::cin >> choice; std::cin.ignore(); // 清除输入缓冲区 switch (choice) { case 1: monitor.displayStatus(); break; case 2: if (monitor.lockBike()) { std::cout << "自行车已锁定" << std::endl; } break; case 3: if (monitor.unlockBike()) { std::cout << "自行车已解锁" << std::endl; } break; case 4: displayGPSTrajectory(); break; case 5: // 在实际实现中,需要发送清除警报命令 std::cout << "警报已清除" << std::endl; break; case 0: appRunning = false; std::cout << "正在退出程序..." << std::endl; break; default: std::cout << "无效选择,请重新输入!" << std::endl; break; } } void displayGPSTrajectory() { auto trajectory = monitor.getGPSTrajectory(); std::cout << "\n=== GPS骑行轨迹 ===" << std::endl; std::cout << "总轨迹点数: " << trajectory.size() << std::endl; for (size_t i = 0; i < trajectory.size(); ++i) { std::cout << "点 " << (i + 1) << ": " << std::fixed << std::setprecision(6) << trajectory[i].first << ", " << trajectory[i].second << std::endl; } } }; // 主函数 int main() { // 设置控制台输出编码为UTF-8,支持中文显示 SetConsoleOutputCP(65001); BikeMonitorApp app; try { app.run(); } catch (const std::exception& e) { std::cerr << "程序异常: " << e.what() << std::endl; return -1; } std::cout << "程序正常退出" << std::endl; return 0; } 模块代码设计#include "stm32f10x.h" // 硬件引脚定义 #define HALL_SENSOR_PIN GPIO_Pin_0 // PA0 - 霍尔传感器 #define BUZZER_PIN GPIO_Pin_1 // PA1 - 蜂鸣器 #define RELAY_PIN GPIO_Pin_2 // PA2 - 继电器控制电磁锁 #define OLED_SCL_PIN GPIO_Pin_6 // PB6 - OLED SCL #define OLED_SDA_PIN GPIO_Pin_7 // PB7 - OLED SDA #define MPU6050_SCL_PIN GPIO_Pin_10 // PB10 - MPU6050 SCL #define MPU6050_SDA_PIN GPIO_Pin_11 // PB11 - MPU6050 SDA #define GPS_TX_PIN GPIO_Pin_2 // PA2 - GPS TX #define GPS_RX_PIN GPIO_Pin_3 // PA3 - GPS RX #define ESP8266_TX_PIN GPIO_Pin_9 // PA9 - ESP8266 TX #define ESP8266_RX_PIN GPIO_Pin_10 // PA10 - ESP8266 RX // MPU6050寄存器地址 #define MPU6050_ADDR 0xD0 #define PWR_MGMT_1 0x6B #define ACCEL_XOUT_H 0x3B #define GYRO_XOUT_H 0x43 // 全局变量 volatile uint32_t wheel_pulse_count = 0; volatile uint32_t total_distance = 0; // 单位:米 volatile float current_speed = 0.0; // 单位:km/h volatile uint8_t theft_alert_flag = 0; // 车轮周长(米),根据实际车轮尺寸调整 #define WHEEL_CIRCUMFERENCE 2.0 // 延时函数 void delay_ms(uint32_t nCount) { for(uint32_t i = 0; i < nCount * 1000; i++); } // GPIO初始化 void GPIO_Init_Config(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN; // 霍尔传感器输入 GPIOA->CRL &= ~(GPIO_CRL_MODE0 | GPIO_CRL_CNF0); GPIOA->CRL |= GPIO_CRL_CNF0_1; // 浮空输入 // 蜂鸣器输出 GPIOA->CRL &= ~(GPIO_CRL_MODE1 | GPIO_CRL_CNF1); GPIOA->CRL |= GPIO_CRL_MODE1_0; // 推挽输出,10MHz // 继电器输出 GPIOA->CRL &= ~(GPIO_CRL_MODE2 | GPIO_CRL_CNF2); GPIOA->CRL |= GPIO_CRL_MODE2_0; // 推挽输出,10MHz // I2C引脚 GPIOB->CRL &= ~(GPIO_CRL_MODE6 | GPIO_CRL_CNF6); GPIOB->CRL |= GPIO_CRL_MODE6_0 | GPIO_CRL_CNF6_1; // 复用开漏输出 GPIOB->CRL &= ~(GPIO_CRL_MODE7 | GPIO_CRL_CNF7); GPIOB->CRL |= GPIO_CRL_MODE7_0 | GPIO_CRL_CNF7_1; // 复用开漏输出 } // 外部中断初始化 - 霍尔传感器 void EXTI_Init_Config(void) { RCC->APB2ENR |= RCC_APB2ENR_AFIOEN; // PA0连接到EXTI0 AFIO->EXTICR[0] &= ~AFIO_EXTICR1_EXTI0; AFIO->EXTICR[0] |= AFIO_EXTICR1_EXTI0_PA; // 上升沿触发 EXTI->RTSR |= EXTI_RTSR_TR0; // 使能EXTI0中断 EXTI->IMR |= EXTI_IMR_MR0; // 配置NVIC NVIC_EnableIRQ(EXTI0_IRQn); NVIC_SetPriority(EXTI0_IRQn, 0x00); } // I2C1初始化 - OLED显示 void I2C1_Init_Config(void) { RCC->APB1ENR |= RCC_APB1ENR_I2C1EN; // 复位I2C1 I2C1->CR1 |= I2C_CR1_SWRST; I2C1->CR1 &= ~I2C_CR1_SWRST; // 配置时钟频率 I2C1->CR2 |= 36; // APB1时钟36MHz I2C1->CCR = 180; // 100kHz标准模式 I2C1->TRISE = 37; // 最大上升时间 // 使能I2C1 I2C1->CR1 |= I2C_CR1_PE; } // I2C2初始化 - MPU6050 void I2C2_Init_Config(void) { RCC->APB1ENR |= RCC_APB1ENR_I2C2EN; // 复位I2C2 I2C2->CR1 |= I2C_CR1_SWRST; I2C2->CR1 &= ~I2C_CR1_SWRST; // 配置时钟频率 I2C2->CR2 |= 36; // APB1时钟36MHz I2C2->CCR = 180; // 100kHz标准模式 I2C2->TRISE = 37; // 最大上升时间 // 使能I2C2 I2C2->CR1 |= I2C_CR1_PE; } // UART1初始化 - ESP8266通信 void UART1_Init_Config(void) { RCC->APB2ENR |= RCC_APB2ENR_USART1EN; // 配置波特率 115200 USART1->BRR = 36000000 / 115200; // 使能发送和接收 USART1->CR1 |= USART_CR1_TE | USART_CR1_RE; // 使能UART1 USART1->CR1 |= USART_CR1_UE; } // UART2初始化 - GPS模块 void UART2_Init_Config(void) { RCC->APB1ENR |= RCC_APB1ENR_USART2EN; // 配置波特率 9600 USART2->BRR = 36000000 / 9600; // 使能发送和接收 USART2->CR1 |= USART_CR1_TE | USART_CR1_RE; // 使能UART2 USART2->CR1 |= USART_CR1_UE; } // I2C起始信号 void I2C_Start(I2C_TypeDef* I2Cx) { I2Cx->CR1 |= I2C_CR1_START; while(!(I2Cx->SR1 & I2C_SR1_SB)); } // I2C停止信号 void I2C_Stop(I2C_TypeDef* I2Cx) { I2Cx->CR1 |= I2C_CR1_STOP; while(I2Cx->SR2 & I2C_SR2_MSL); } // I2C发送地址 void I2C_SendAddress(I2C_TypeDef* I2Cx, uint8_t address) { I2Cx->DR = address; while(!(I2Cx->SR1 & I2C_SR1_ADDR)); (void)I2Cx->SR2; // 读取SR2清除ADDR标志 } // I2C发送数据 void I2C_SendData(I2C_TypeDef* I2Cx, uint8_t data) { I2Cx->DR = data; while(!(I2Cx->SR1 & I2C_SR1_TXE)); } // I2C接收数据 uint8_t I2C_ReceiveData(I2C_TypeDef* I2Cx) { while(!(I2Cx->SR1 & I2C_SR1_RXNE)); return I2Cx->DR; } // MPU6050初始化 void MPU6050_Init(void) { // 唤醒MPU6050 I2C_Start(I2C2); I2C_SendAddress(I2C2, MPU6050_ADDR); I2C_SendData(I2C2, PWR_MGMT_1); I2C_SendData(I2C2, 0x00); // 解除休眠 I2C_Stop(I2C2); delay_ms(10); } // 读取MPU6050加速度数据 void MPU6050_ReadAccel(int16_t* accel_data) { I2C_Start(I2C2); I2C_SendAddress(I2C2, MPU6050_ADDR); I2C_SendData(I2C2, ACCEL_XOUT_H); I2C_Start(I2C2); I2C_SendAddress(I2C2, MPU6050_ADDR | 0x01); for(uint8_t i = 0; i < 6; i++) { if(i == 5) { I2C2->CR1 &= ~I2C_CR1_ACK; // 最后一个字节不发送ACK } ((uint8_t*)accel_data)[i] = I2C_ReceiveData(I2C2); } I2C_Stop(I2C2); } // 蜂鸣器控制 void Buzzer_Control(uint8_t state) { if(state) { GPIOA->BSRR = BUZZER_PIN; // 置位 } else { GPIOA->BRR = BUZZER_PIN; // 复位 } } // 电磁锁控制 void ElectromagneticLock_Control(uint8_t state) { if(state) { GPIOA->BSRR = RELAY_PIN; // 锁定 } else { GPIOA->BRR = RELAY_PIN; // 解锁 } } // OLED显示初始化(简化版) void OLED_Init(void) { // OLED初始化序列 uint8_t init_cmd[] = {0xAE, 0x20, 0x10, 0xB0, 0xC8, 0x00, 0x10, 0x40, 0x81, 0xFF, 0xA1, 0xA6, 0xA8, 0x3F, 0xA4, 0xD3, 0x00, 0xD5, 0xF0, 0xD9, 0x22, 0xDA, 0x12, 0xDB, 0x20, 0x8D, 0x14, 0xAF}; for(uint8_t i = 0; i < sizeof(init_cmd); i++) { I2C_Start(I2C1); I2C_SendAddress(I2C1, 0x78); // OLED地址 I2C_SendData(I2C1, 0x00); // 命令模式 I2C_SendData(I2C1, init_cmd[i]); I2C_Stop(I2C1); } } // OLED显示字符串(简化版) void OLED_ShowString(uint8_t x, uint8_t y, char* str) { // 设置显示位置 I2C_Start(I2C1); I2C_SendAddress(I2C1, 0x78); I2C_SendData(I2C1, 0x00); I2C_SendData(I2C1, 0xB0 + y); I2C_SendData(I2C1, 0x00); I2C_SendData(I2C1, 0x10 + (x & 0x0F)); I2C_SendData(I2C1, 0x00); I2C_SendData(I2C1, 0x01 + ((x >> 4) & 0x0F)); I2C_Stop(I2C1); // 发送字符数据 while(*str) { I2C_Start(I2C1); I2C_SendAddress(I2C1, 0x78); I2C_SendData(I2C1, 0x40); // 数据模式 // 这里简化字符显示,实际需要字库 I2C_SendData(I2C1, *str++); I2C_Stop(I2C1); } } // 速度计算函数 void Calculate_Speed(void) { static uint32_t last_pulse_count = 0; static uint32_t last_time = 0; uint32_t current_time = SysTick->VAL; if(current_time < last_time) { uint32_t pulse_diff = wheel_pulse_count - last_pulse_count; uint32_t time_diff = last_time - current_time; // 计算速度 (km/h) current_speed = (pulse_diff * WHEEL_CIRCUMFERENCE * 3.6f) / (time_diff * 0.001f); // 计算总里程 total_distance += pulse_diff * WHEEL_CIRCUMFERENCE; last_pulse_count = wheel_pulse_count; } last_time = current_time; } // 防盗检测函数 void Theft_Detection(void) { int16_t accel_data[3]; MPU6050_ReadAccel(accel_data); // 计算加速度矢量大小 float accel_magnitude = sqrt(accel_data[0]*accel_data[0] + accel_data[1]*accel_data[1] + accel_data[2]*accel_data[2]); // 如果检测到异常移动且自行车未在骑行状态 if(accel_magnitude > 2000 && current_speed < 1.0) { theft_alert_flag = 1; ElectromagneticLock_Control(1); // 锁定电磁锁 Buzzer_Control(1); // 触发蜂鸣器 // 通过ESP8266发送警报 USART1->DR = 'A'; // 发送警报信号 while(!(USART1->SR & USART_SR_TC)); } } // 系统初始化 void System_Init(void) { // 系统时钟配置 RCC->CR |= RCC_CR_HSEON; while(!(RCC->CR & RCC_CR_HSERDY)); RCC->CFGR |= RCC_CFGR_PLLMULL9; RCC->CFGR |= RCC_CFGR_PLLSRC; RCC->CR |= RCC_CR_PLLON; while(!(RCC->CR & RCC_CR_PLLRDY)); RCC->CFGR |= RCC_CFGR_SW_PLL; while(!(RCC->CFGR & RCC_CFGR_SWS_PLL)); // 各模块初始化 GPIO_Init_Config(); EXTI_Init_Config(); I2C1_Init_Config(); I2C2_Init_Config(); UART1_Init_Config(); UART2_Init_Config(); MPU6050_Init(); OLED_Init(); // 配置SysTick定时器 SysTick->LOAD = 9000 - 1; // 1ms中断 SysTick->VAL = 0; SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk | SysTick_CTRL_ENABLE_Msk; } // 霍尔传感器中断服务函数 void EXTI0_IRQHandler(void) { if(EXTI->PR & EXTI_PR_PR0) { wheel_pulse_count++; EXTI->PR = EXTI_PR_PR0; // 清除中断标志 } } // 主函数 int main(void) { System_Init(); char display_buffer[20]; while(1) { // 计算速度 Calculate_Speed(); // 防盗检测 Theft_Detection(); // OLED显示更新 sprintf(display_buffer, "Speed:%.1fkm/h", current_speed); OLED_ShowString(0, 0, display_buffer); sprintf(display_buffer, "Dist:%lum", total_distance); OLED_ShowString(0, 2, display_buffer); if(theft_alert_flag) { OLED_ShowString(0, 4, "ALERT! THEFT!"); } else { OLED_ShowString(0, 4, "Status:Normal"); } delay_ms(500); } } // UART1中断服务函数 - ESP8266数据接收 void USART1_IRQHandler(void) { if(USART1->SR & USART_SR_RXNE) { uint8_t data = USART1->DR; // 处理来自手机APP的指令 if(data == 'U') { // 解锁指令 ElectromagneticLock_Control(0); theft_alert_flag = 0; Buzzer_Control(0); } } } // UART2中断服务函数 - GPS数据接收 void USART2_IRQHandler(void) { if(USART2->SR & USART_SR_RXNE) { uint8_t gps_data = USART2->DR; // 解析GPS数据(NMEA协议) // 这里简化处理,实际需要完整解析 } } 项目核心代码#include "stm32f10x.h" #include "oled.h" #include "mpu6050.h" #include "gps.h" #include "esp8266.h" #include "hall_sensor.h" #include "buzzer.h" #include "lock.h" // 全局变量定义 volatile uint32_t hall_count = 0; volatile uint32_t total_distance = 0; volatile uint8_t speed_kmh = 0; volatile uint8_t theft_alarm = 0; volatile uint8_t system_locked = 0; // 函数声明 void System_Init(void); void Timer2_Init(void); void EXTI0_Init(void); void USART1_Init(void); void USART2_Init(void); void I2C1_Init(void); void GPIO_Init(void); void Calculate_Speed_Distance(void); void Check_Theft_Status(void); void Update_Display(void); void Send_Alarm_Message(void); int main(void) { // 系统初始化 System_Init(); // 外设初始化 Timer2_Init(); EXTI0_Init(); USART1_Init(); // GPS USART2_Init(); // ESP8266 I2C1_Init(); // MPU6050 & OLED GPIO_Init(); // 模块初始化 OLED_Init(); MPU6050_Init(); GPS_Init(); ESP8266_Init(); Buzzer_Init(); Lock_Init(); // 显示欢迎界面 OLED_ShowString(0, 0, "Bike Monitor", 16); OLED_ShowString(0, 2, "System Ready", 16); Delay_ms(2000); OLED_Clear(); while(1) { // 计算速度和里程 Calculate_Speed_Distance(); // 检测防盗状态 Check_Theft_Status(); // 更新显示 Update_Display(); // 如果检测到异常移动,发送警报 if(theft_alarm) { Send_Alarm_Message(); theft_alarm = 0; } Delay_ms(100); } } // 系统初始化 void System_Init(void) { // 开启时钟 RCC->APB2ENR |= (1<<2)|(1<<3)|(1<<4)|(1<<5)|(1<<8)|(1<<14); RCC->APB1ENR |= (1<<0)|(1<<17)|(1<<21); // 设置系统时钟 SystemInit(); SystemCoreClockUpdate(); } // 定时器2初始化 - 用于速度计算 void Timer2_Init(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); TIM_TimeBaseStructure.TIM_Period = 999; TIM_TimeBaseStructure.TIM_Prescaler = 7199; TIM_TimeBaseStructure.TIM_ClockDivision = 0; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); TIM_Cmd(TIM2, ENABLE); NVIC_EnableIRQ(TIM2_IRQn); } // 外部中断初始化 - 霍尔传感器 void EXTI0_Init(void) { EXTI_InitTypeDef EXTI_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE); GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0); EXTI_InitStructure.EXTI_Line = EXTI_Line0; EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; EXTI_InitStructure.EXTI_LineCmd = ENABLE; EXTI_Init(&EXTI_InitStructure); NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x00; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x00; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } // USART1初始化 - GPS void USART1_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE); // TX - PA9, RX - PA10 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOA, &GPIO_InitStructure); USART_InitStructure.USART_BaudRate = 9600; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART1, &USART_InitStructure); USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); USART_Cmd(USART1, ENABLE); NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } // USART2初始化 - ESP8266 void USART2_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE); // TX - PA2, RX - PA3 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOA, &GPIO_InitStructure); USART_InitStructure.USART_BaudRate = 115200; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART2, &USART_InitStructure); USART_Cmd(USART2, ENABLE); } // I2C1初始化 - MPU6050 & OLED void I2C1_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; I2C_InitTypeDef I2C_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE); // SCL - PB6, SDA - PB7 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD; GPIO_Init(GPIOB, &GPIO_InitStructure); I2C_InitStructure.I2C_Mode = I2C_Mode_I2C; I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2; I2C_InitStructure.I2C_OwnAddress1 = 0x00; I2C_InitStructure.I2C_Ack = I2C_Ack_Enable; I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; I2C_InitStructure.I2C_ClockSpeed = 400000; I2C_Init(I2C1, &I2C_InitStructure); I2C_Cmd(I2C1, ENABLE); } // GPIO初始化 void GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; // 霍尔传感器输入 - PA0 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; GPIO_Init(GPIOA, &GPIO_InitStructure); // 蜂鸣器输出 - PA1 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); // 电磁锁控制 - PA4 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4; GPIO_Init(GPIOA, &GPIO_InitStructure); } // 计算速度和里程 void Calculate_Speed_Distance(void) { static uint32_t last_hall_count = 0; uint32_t current_count = hall_count; // 计算速度 (km/h) // 假设车轮周长2米,每秒脉冲数 * 2 * 3.6 = km/h speed_kmh = (current_count - last_hall_count) * 2 * 3.6; // 计算总里程 (米) total_distance = current_count * 2; last_hall_count = current_count; } // 检测防盗状态 void Check_Theft_Status(void) { static uint8_t stable_count = 0; MPU6050_Data mpu_data; MPU6050_Read_Data(&mpu_data); // 检测异常移动 - 角度变化过大 if(abs(mpu_data.Angle_X) > 30 || abs(mpu_data.Angle_Y) > 30) { stable_count++; if(stable_count > 5) // 连续5次检测到异常 { if(!system_locked) { theft_alarm = 1; system_locked = 1; Lock_Activate(); // 锁定电子锁 Buzzer_On(); // 开启蜂鸣器报警 } } } else { stable_count = 0; } } // 更新显示 void Update_Display(void) { char display_buf[20]; OLED_Clear(); // 显示速度 sprintf(display_buf, "Speed: %d km/h", speed_kmh); OLED_ShowString(0, 0, display_buf, 16); // 显示里程 sprintf(display_buf, "Dist: %ld m", total_distance); OLED_ShowString(0, 2, display_buf, 16); // 显示状态 if(system_locked) { OLED_ShowString(0, 4, "Status: LOCKED", 16); } else { OLED_ShowString(0, 4, "Status: NORMAL", 16); } // 显示GPS状态 if(GPS_Get_Fix_Status()) { OLED_ShowString(0, 6, "GPS: Fixed", 16); } else { OLED_ShowString(0, 6, "GPS: Searching", 16); } } // 发送警报消息 void Send_Alarm_Message(void) { char alarm_msg[100]; GPS_Data gps_data; GPS_Get_Data(&gps_data); sprintf(alarm_msg, "ALERT: Bike theft detected! Location: Lat:%f,Lon:%f", gps_data.latitude, gps_data.longitude); ESP8266_Send_Message(alarm_msg); } // 定时器2中断服务函数 - 每秒执行一次 void TIM2_IRQHandler(void) { if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) { TIM_ClearITPendingBit(TIM2, TIM_IT_Update); // 每秒更新一次GPS数据 GPS_Update(); } } // 外部中断0服务函数 - 霍尔传感器计数 void EXTI0_IRQHandler(void) { if(EXTI_GetITStatus(EXTI_Line0) != RESET) { hall_count++; EXTI_ClearITPendingBit(EXTI_Line0); } } // USART1中断服务函数 - GPS数据接收 void USART1_IRQHandler(void) { if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { uint8_t data = USART_ReceiveData(USART1); GPS_Receive_Byte(data); USART_ClearITPendingBit(USART1, USART_IT_RXNE); } } // 延时函数 void Delay_ms(uint32_t nCount) { volatile uint32_t i, j; for(i = 0; i < nCount; i++) for(j = 0; j < 8000; j++); } 总结智能自行车状态监测与防盗系统通过集成多种传感器和通信模块,实现了对自行车状态的全面监控与高效防盗。该系统能够实时计算速度和里程,监测车身姿态变化,并在检测到异常移动时自动锁定电子锁并向用户发送警报,同时记录骑行轨迹供用户随时查看。硬件部分以STM32F103C8T6单片机为核心,结合霍尔传感器、MPU6050姿态传感器和GPS模块进行数据采集与定位,并通过ESP8266等通信模块实现远程警报功能。所有数据可在本地OLED屏幕上直观显示,确保系统操作简便且响应迅速。整体而言,这一系统将传感技术、通信控制与用户交互紧密结合,不仅提升了自行车的安全性,还为用户提供了便捷的状态跟踪体验,具有较高的实用性和可靠性。
  • [技术干货] 实验室紫外灯智能消毒机器人
    项目开发背景实验室环境通常涉及各种生物或化学实验,存在较高的微生物污染风险,尤其是细菌、病毒等病原体可能通过空气或表面传播,威胁实验人员的健康与实验结果的准确性。传统消毒方式如人工喷洒消毒液或固定紫外灯照射,往往效率低下且覆盖不均,难以确保实验室每个角落都得到彻底清洁。此外,紫外灯在使用时若未及时关闭,可能对误入人员造成伤害,因此亟需一种智能化、自动化的解决方案来提升消毒效率和安全性。紫外灯智能消毒机器人的开发正是基于这一需求,旨在通过自动化技术实现高效、安全的实验室环境维护。该机器人能够自主移动并覆盖预设路径,确保消毒无死角,同时结合红外人体感应模块,在检测到人员活动时自动暂停紫外灯工作,避免潜在的健康风险。这种设计不仅减少了人工干预,还显著降低了因疏忽导致的安全事故,符合现代实验室对智能化和精细化管理的要求。在功能实现上,机器人通过定时设置和自动返回充电坞的设计,进一步提升了操作的便捷性和持续性。防跌落传感器的加入则确保了机器人在复杂实验室环境中的稳定运行,防止意外跌落造成的设备损坏或安全隐患。这些特性共同构成了一个可靠的消毒系统,能够适应实验室的高标准卫生需求。硬件模块的集成,如STM32主控芯片和电机驱动系统,为机器人的精准控制和高效执行提供了基础支撑。通过将移动底盘、紫外灯执行器与安全传感器有机结合,该项目不仅推动了实验室消毒技术的升级,还为未来智能机器人在其他领域的应用积累了宝贵经验。总体而言,该开发项目旨在通过创新技术解决现实问题,促进实验室管理的智能化和人性化发展。设计实现的功能(1) 使用STM32F103C8T6单片机控制L298N电机驱动板,驱动带编码器的直流减速电机,实现机器人底盘按照预设路径在实验室内移动。(2) 通过HC-SR501人体红外传感器检测人体存在,STM32在确认无人状态下控制继电器模块开启紫外灯管进行消毒。(3) STM32具备定时功能,可设置消毒时长,时间结束后控制机器人自动返回充电坞并关闭紫外灯。(4) 使用红外对管防跌落传感器,STM32实时监测,在检测到跌落风险时采取安全措施,如停止移动。项目硬件模块组成(1)主控芯片:STM32F103C8T6单片机。(2)移动底盘:带编码器的直流减速电机与车轮、L298N电机驱动板。(3)核心执行器:紫外灯管与继电器模块。(4)安全传感:HC-SR501人体红外传感器、红外对管防跌落传感器。设计意义该实验室紫外灯智能消毒机器人的设计意义在于实现自动化消毒流程,通过预设路径移动功能,机器人能够自主覆盖实验室空间,无需人工干预即可完成消毒任务,显著提升消毒效率并降低人力成本。同时,机器人集成了红外人体感应模块,确保仅在无人状态下开启紫外灯,有效避免紫外线辐射对人员的潜在危害,保障了实验室操作的安全性。防跌落功能通过红外对管传感器实时监测环境,防止机器人在运行中发生意外跌落,进一步增强了设备的可靠性和使用寿命。定时功能与自动返回充电坞的设计,使得机器人能够精确控制消毒时长并在结束后自主归位,不仅优化了能源管理,还确保了消毒过程的规范化和连续性。整体而言,该设计结合了移动底盘、智能传感和核心执行器,为实验室环境提供了一种高效、安全的消毒解决方案,有助于维持洁净的实验条件,减少交叉污染风险,并支持科研工作的顺利开展。设计思路该机器人的设计以STM32F103C8T6单片机作为核心控制器,负责协调各硬件模块的运行。通过预设程序控制底盘移动、安全检测和消毒执行,确保整个系统高效可靠地工作。移动底盘采用带编码器的直流减速电机和L298N电机驱动板,实现精确的速度和位置控制。机器人能够按照预设路径在实验室内自主移动,编码器反馈数据用于校正路径偏差,保证移动的准确性。安全方面,HC-SR501人体红外传感器实时监测环境,仅在检测到无人状态时才允许开启紫外灯,避免对人体造成伤害。同时,红外对管防跌落传感器安装在机器人底部,当检测到边缘或障碍时立即停止移动,防止跌落事故。消毒功能通过继电器模块控制紫外灯管的开关,结合定时设置实现自动化操作。用户可预设消毒时长,结束后系统自动关闭紫外灯,并引导机器人返回充电坞进行充电,确保能源管理和任务连续性。整体设计注重实用性和安全性,各模块通过主控芯片集成,实现移动、检测和消毒的协同工作,无需人工干预即可完成实验室的智能消毒任务。框架图+-------------------+ +-----------------------+ | | | | | 人体红外传感器 |------>| | | (HC-SR501) | | | +-------------------+ | | | 主控芯片 | +-------------------+ | (STM32F103C8T6) | | |------>| | | 防跌落传感器 | | | | (红外对管) | | | +-------------------+ +-----------------------+ | | | | +-------------------+ | | | |<------| | | 电机驱动板 | | | | (L298N) | | | +-------------------+ | | | | | v | | +-------------------+ | | | 直流减速电机 | | | | (带编码器) |------>| | +-------------------+ | | | | +-------------------+ | | | |<------| | | 继电器模块 | | | +-------------------+ | | | +-----------------------+ v +-------------------+ | 紫外灯管 | +-------------------+ 系统总体设计系统总体设计以STM32F103C8T6单片机作为核心控制单元,负责协调和管理机器人的各项功能。移动底盘采用带编码器的直流减速电机与L298N电机驱动板组合,实现机器人在实验室内按照预设路径自主移动,编码器提供位置反馈以确保移动精度。消毒功能通过HC-SR501人体红外传感器检测环境状态,仅在确认无人时才会通过继电器模块开启紫外灯管进行消毒。系统集成定时功能,允许用户设置消毒时长,消毒结束后机器人自动返回充电坞并关闭紫外灯,实现全自动化操作。安全方面,系统配备红外对管防跌落传感器,实时监测机器人周围环境,防止在移动过程中从高处跌落,确保运行过程的稳定性和安全性。整体设计注重实际应用,各硬件模块协同工作,满足实验室消毒的可靠需求。系统功能总结编号功能描述实现硬件1机器人底盘可按照预设路径在实验室内移动。移动底盘:带编码器的直流减速电机与车轮、L298N电机驱动板2通过红外人体感应模块确保在无人状态下才开启紫外灯进行消毒。HC-SR501人体红外传感器、紫外灯管与继电器模块3具备定时功能,可设置消毒时长,结束后自动返回充电坞并关闭紫外灯。主控芯片STM32F103C8T6(控制定时和逻辑)、移动底盘(返回动作)4具备防跌落功能,确保运行安全。红外对管防跌落传感器设计的各个功能模块描述主控芯片模块基于STM32F103C8T6单片机,作为整个系统的控制核心,负责协调各模块的运行。它处理来自传感器的输入信号,控制移动底盘按照预设路径移动,并管理定时功能,实现在设置消毒时长结束后自动返回充电坞并关闭紫外灯。同时,该芯片通过逻辑判断确保仅在无人状态下才允许开启紫外灯,以保障安全。移动底盘模块由带编码器的直流减速电机、车轮和L298N电机驱动板组成,实现机器人在实验室内的自主移动。编码器提供位置反馈,帮助精确控制移动轨迹,而L298N驱动板调节电机的转速和方向,确保底盘能够稳定地沿预设路径运行,并在需要时返回充电坞。核心执行器模块包括紫外灯管和继电器模块,紫外灯管用于进行消毒操作,继电器模块在主控芯片的控制下开关紫外灯。该模块仅在红外人体感应模块确认无人状态时被激活,从而避免对人员造成伤害,并在定时结束后自动关闭紫外灯。安全传感模块由HC-SR501人体红外传感器和红外对管防跌落传感器构成。人体红外传感器检测实验室内是否有人存在,确保紫外灯只在无人环境下开启;防跌落传感器通过红外对管监测地面边缘或障碍物,防止机器人在移动过程中发生跌落,保障运行安全。上位机代码设计// main.cpp #include <QApplication> #include <QMainWindow> #include <QVBoxLayout> #include <QHBoxLayout> #include <QPushButton> #include <QLabel> #include <QLineEdit> #include <QTextEdit> #include <QGroupBox> #include <QSerialPort> #include <QSerialPortInfo> #include <QTimer> #include <QMessageBox> #include <QProgressBar> #include <QSpinBox> #include <QCheckBox> #include <QDateTime> class RobotController : public QMainWindow { Q_OBJECT public: RobotController(QWidget *parent = nullptr) : QMainWindow(parent) { setupUI(); setupSerialPort(); setupConnections(); } ~RobotController() { if(serialPort->isOpen()) serialPort->close(); } private slots: void connectRobot() { serialPort->setPortName(portComboBox->currentText()); serialPort->setBaudRate(QSerialPort::Baud115200); if(serialPort->open(QIODevice::ReadWrite)) { statusLabel->setText("已连接到机器人"); logTextEdit->append("[" + QDateTime::currentDateTime().toString("hh:mm:ss") + "] 机器人连接成功"); } else { QMessageBox::warning(this, "连接失败", "无法连接到串口设备"); } } void disconnectRobot() { if(serialPort->isOpen()) { serialPort->close(); statusLabel->setText("未连接"); logTextEdit->append("[" + QDateTime::currentDateTime().toString("hh:mm:ss") + "] 断开连接"); } } void startDisinfection() { if(!serialPort->isOpen()) { QMessageBox::warning(this, "错误", "请先连接机器人"); return; } int duration = durationSpinBox->value(); QString command = QString("START:%1").arg(duration); serialPort->write(command.toUtf8()); logTextEdit->append("[" + QDateTime::currentDateTime().toString("hh:mm:ss") + "] 开始消毒,时长: " + QString::number(duration) + "分钟"); // 启动进度条更新 progressBar->setMaximum(duration * 60); // 转换为秒 progressBar->setValue(0); // 启动定时器更新进度 progressTimer->start(1000); } void stopDisinfection() { if(serialPort->isOpen()) { serialPort->write("STOP"); logTextEdit->append("[" + QDateTime::currentDateTime().toString("hh:mm:ss") + "] 停止消毒"); progressTimer->stop(); progressBar->setValue(0); } } void returnToDock() { if(serialPort->isOpen()) { serialPort->write("RETURN"); logTextEdit->append("[" + QDateTime::currentDateTime().toString("hh:mm:ss") + "] 返回充电坞"); } } void updateProgress() { int current = progressBar->value() + 1; progressBar->setValue(current); if(current >= progressBar->maximum()) { progressTimer->stop(); logTextEdit->append("[" + QDateTime::currentDateTime().toString("hh:mm:ss") + "] 消毒完成"); } } void readSerialData() { QByteArray data = serialPort->readAll(); QString message = QString::fromUtf8(data); if(!message.isEmpty()) { logTextEdit->append("机器人: " + message.trimmed()); if(message.contains("HUMAN_DETECTED")) { QMessageBox::warning(this, "安全警告", "检测到人员活动,紫外灯已关闭"); } else if(message.contains("DISINFECTION_COMPLETE")) { progressTimer->stop(); progressBar->setValue(progressBar->maximum()); logTextEdit->append("[" + QDateTime::currentDateTime().toString("hh:mm:ss") + "] 消毒任务完成"); } } } void refreshPorts() { portComboBox->clear(); QList<QSerialPortInfo> ports = QSerialPortInfo::availablePorts(); for(const QSerialPortInfo &port : ports) { portComboBox->addItem(port.portName()); } } private: void setupUI() { setWindowTitle("实验室紫外灯智能消毒机器人控制系统"); setFixedSize(600, 700); QWidget *centralWidget = new QWidget(this); setCentralWidget(centralWidget); QVBoxLayout *mainLayout = new QVBoxLayout(centralWidget); // 连接设置组 QGroupBox *connectionGroup = new QGroupBox("连接设置"); QHBoxLayout *connectionLayout = new QHBoxLayout(connectionGroup); portComboBox = new QComboBox(); refreshButton = new QPushButton("刷新端口"); connectButton = new QPushButton("连接"); disconnectButton = new QPushButton("断开"); connectionLayout->addWidget(new QLabel("串口:")); connectionLayout->addWidget(portComboBox); connectionLayout->addWidget(refreshButton); connectionLayout->addWidget(connectButton); connectionLayout->addWidget(disconnectButton); // 控制组 QGroupBox *controlGroup = new QGroupBox("机器人控制"); QVBoxLayout *controlLayout = new QVBoxLayout(controlGroup); QHBoxLayout *durationLayout = new QHBoxLayout(); durationLayout->addWidget(new QLabel("消毒时长(分钟):")); durationSpinBox = new QSpinBox(); durationSpinBox->setRange(1, 120); durationSpinBox->setValue(30); durationLayout->addWidget(durationSpinBox); durationLayout->addStretch(); startButton = new QPushButton("开始消毒"); stopButton = new QPushButton("停止消毒"); returnButton = new QPushButton("返回充电坞"); QHBoxLayout *buttonLayout = new QHBoxLayout(); buttonLayout->addWidget(startButton); buttonLayout->addWidget(stopButton); buttonLayout->addWidget(returnButton); progressBar = new QProgressBar(); controlLayout->addLayout(durationLayout); controlLayout->addLayout(buttonLayout); controlLayout->addWidget(progressBar); // 状态显示组 QGroupBox *statusGroup = new QGroupBox("系统状态"); QVBoxLayout *statusLayout = new QVBoxLayout(statusGroup); statusLabel = new QLabel("未连接"); logTextEdit = new QTextEdit(); logTextEdit->setReadOnly(true); statusLayout->addWidget(statusLabel); statusLayout->addWidget(logTextEdit); // 添加到主布局 mainLayout->addWidget(connectionGroup); mainLayout->addWidget(controlGroup); mainLayout->addWidget(statusGroup); refreshPorts(); } void setupSerialPort() { serialPort = new QSerialPort(this); progressTimer = new QTimer(this); } void setupConnections() { connect(connectButton, &QPushButton::clicked, this, &RobotController::connectRobot); connect(disconnectButton, &QPushButton::clicked, this, &RobotController::disconnectRobot); connect(refreshButton, &QPushButton::clicked, this, &RobotController::refreshPorts); connect(startButton, &QPushButton::clicked, this, &RobotController::startDisinfection); connect(stopButton, &QPushButton::clicked, this, &RobotController::stopDisinfection); connect(returnButton, &QPushButton::clicked, this, &RobotController::returnToDock); connect(serialPort, &QSerialPort::readyRead, this, &RobotController::readSerialData); connect(progressTimer, &QTimer::timeout, this, &RobotController::updateProgress); } // UI组件 QComboBox *portComboBox; QPushButton *refreshButton; QPushButton *connectButton; QPushButton *disconnectButton; QPushButton *startButton; QPushButton *stopButton; QPushButton *returnButton; QSpinBox *durationSpinBox; QProgressBar *progressBar; QLabel *statusLabel; QTextEdit *logTextEdit; // 功能组件 QSerialPort *serialPort; QTimer *progressTimer; }; int main(int argc, char *argv[]) { QApplication app(argc, argv); RobotController controller; controller.show(); return app.exec(); } #include "main.moc" # RobotController.pro QT += core gui serialport greaterThan(QT_MAJOR_VERSION, 4): QT += widgets TARGET = RobotController TEMPLATE = app SOURCES += main.cpp这个上位机程序提供了以下功能:串口通信:通过串口与STM32机器人通信连接管理:自动检测可用串口,支持连接/断开操作消毒控制:设置消毒时长(1-120分钟)开始/停止消毒手动返回充电坞状态监控:实时显示机器人状态信息消毒进度条显示安全警告(检测到人体活动)日志记录:所有操作和机器人反馈都有时间戳记录通信协议说明:START:时长 - 开始消毒STOP - 停止消毒RETURN - 返回充电坞HUMAN_DETECTED - 机器人检测到人员DISINFECTION_COMPLETE - 消毒完成模块代码设计#include <stdint.h> // 寄存器地址定义 #define RCC_BASE 0x40021000 #define GPIOA_BASE 0x40010800 #define GPIOB_BASE 0x40010C00 #define GPIOC_BASE 0x40011000 #define TIM2_BASE 0x40000000 #define TIM3_BASE 0x40000400 #define TIM4_BASE 0x40000800 #define NVIC_BASE 0xE000E100 #define EXTI_BASE 0x40010400 #define AFIO_BASE 0x40010000 // RCC 寄存器 #define RCC_APB2ENR *(volatile uint32_t *)(RCC_BASE + 0x18) #define RCC_APB1ENR *(volatile uint32_t *)(RCC_BASE + 0x1C) // GPIO 寄存器结构 typedef struct { volatile uint32_t CRL; volatile uint32_t CRH; volatile uint32_t IDR; volatile uint32_t ODR; volatile uint32_t BSRR; volatile uint32_t BRR; volatile uint32_t LCKR; } GPIO_TypeDef; #define GPIOA ((GPIO_TypeDef *)GPIOA_BASE) #define GPIOB ((GPIO_TypeDef *)GPIOB_BASE) #define GPIOC ((GPIO_TypeDef *)GPIOC_BASE) // 定时器寄存器结构 typedef struct { volatile uint32_t CR1; volatile uint32_t CR2; volatile uint32_t SMCR; volatile uint32_t DIER; volatile uint32_t SR; volatile uint32_t EGR; volatile uint32_t CCMR1; volatile uint32_t CCMR2; volatile uint32_t CCER; volatile uint32_t CNT; volatile uint32_t PSC; volatile uint32_t ARR; volatile uint32_t CCR1; volatile uint32_t CCR2; volatile uint32_t CCR3; volatile uint32_t CCR4; } TIM_TypeDef; #define TIM2 ((TIM_TypeDef *)TIM2_BASE) #define TIM3 ((TIM_TypeDef *)TIM3_BASE) #define TIM4 ((TIM_TypeDef *)TIM4_BASE) // NVIC 寄存器 #define NVIC_ISER0 *(volatile uint32_t *)(NVIC_BASE + 0x00) // EXTI 寄存器 #define EXTI_IMR *(volatile uint32_t *)(EXTI_BASE + 0x00) #define EXTI_RTSR *(volatile uint32_t *)(EXTI_BASE + 0x08) #define EXTI_FTSR *(volatile uint32_t *)(EXTI_BASE + 0x0C) #define EXTI_PR *(volatile uint32_t *)(EXTI_BASE + 0x14) // AFIO 寄存器 #define AFIO_EXTICR1 *(volatile uint32_t *)(AFIO_BASE + 0x08) // 引脚定义 #define MOTOR_A_IN1_PIN 0 // PA0 #define MOTOR_A_IN2_PIN 1 // PA1 #define MOTOR_B_IN3_PIN 2 // PA2 #define MOTOR_B_IN4_PIN 3 // PA3 #define UV_RELAY_PIN 13 // PC13 #define HUMAN_IR_PIN 0 // PB0 #define DROP_IR_PIN 1 // PB1 #define ENCODER_A_PIN1 6 // PB6 (TIM4_CH1) #define ENCODER_A_PIN2 7 // PB7 (TIM4_CH2) // 全局变量 volatile uint8_t human_detected = 0; volatile uint8_t drop_detected = 0; volatile uint32_t disinfect_time_remaining = 0; volatile uint8_t disinfect_active = 0; // 函数声明 void SystemInit(void); void GPIO_Init(void); void TIM_Init(void); void NVIC_Init(void); void Motor_Control(uint8_t motor, int8_t speed); void UV_Light_Control(uint8_t state); uint8_t Read_Human_Sensor(void); uint8_t Read_Drop_Sensor(void); void Disinfect_Timer_Start(uint32_t duration); void Disinfect_Timer_Stop(void); void Move_Preset_Path(void); void TIM2_IRQHandler(void) __attribute__((interrupt)); void EXTI0_IRQHandler(void) __attribute__((interrupt)); void EXTI1_IRQHandler(void) __attribute__((interrupt)); // 系统初始化 void SystemInit(void) { // 启用时钟 RCC_APB2ENR |= (1 << 2) | (1 << 3) | (1 << 4) | (1 << 0); // GPIOA, GPIOB, GPIOC, AFIO RCC_APB1ENR |= (1 << 0) | (1 << 1) | (1 << 2); // TIM2, TIM3, TIM4 } // GPIO 初始化 void GPIO_Init(void) { // GPIOA 配置: MOTOR_A_IN1, MOTOR_A_IN2, MOTOR_B_IN3, MOTOR_B_IN4 为推挽输出 GPIOA->CRL &= ~(0xFF << (MOTOR_A_IN1_PIN * 4)); // 清除配置 GPIOA->CRL |= (0x01 << (MOTOR_A_IN1_PIN * 4)) | (0x01 << (MOTOR_A_IN2_PIN * 4)) | (0x01 << (MOTOR_B_IN3_PIN * 4)) | (0x01 << (MOTOR_B_IN4_PIN * 4)); // 推挽输出,50MHz // GPIOB 配置: HUMAN_IR_PIN 和 DROP_IR_PIN 为输入上拉,ENCODER_A_PIN1 和 ENCODER_A_PIN2 为输入 GPIOB->CRL &= ~(0xFF << (HUMAN_IR_PIN * 4)); // 清除 HUMAN_IR_PIN 和 DROP_IR_PIN 配置 GPIOB->CRL |= (0x08 << (HUMAN_IR_PIN * 4)) | (0x08 << (DROP_IR_PIN * 4)); // 输入上拉 GPIOB->CRL &= ~(0xFF << (ENCODER_A_PIN1 * 4)); // 清除编码器引脚配置 GPIOB->CRL |= (0x04 << (ENCODER_A_PIN1 * 4)) | (0x04 << (ENCODER_A_PIN2 * 4)); // 输入浮空 // GPIOC 配置: UV_RELAY_PIN 为推挽输出 GPIOC->CRH &= ~(0xF << ((UV_RELAY_PIN - 8) * 4)); // 清除配置 GPIOC->CRH |= (0x01 << ((UV_RELAY_PIN - 8) * 4)); // 推挽输出,50MHz // 设置上拉电阻 for HUMAN_IR_PIN and DROP_IR_PIN GPIOB->ODR |= (1 << HUMAN_IR_PIN) | (1 << DROP_IR_PIN); } // 定时器初始化 void TIM_Init(void) { // TIM3 用于电机 PWM TIM3->PSC = 71; // 预分频器,72MHz / (71+1) = 1MHz TIM3->ARR = 999; // 自动重载值,1MHz / 1000 = 1kHz PWM TIM3->CCMR1 |= (0x06 << 4) | (0x06 << 12); // CH1 和 CH2 PWM 模式 1 TIM3->CCER |= (1 << 0) | (1 << 4); // 启用 CH1 和 CH2 TIM3->CR1 |= (1 << 0); // 启用 TIM3 // TIM4 用于编码器接口 TIM4->PSC = 0; TIM4->ARR = 0xFFFF; TIM4->SMCR |= (0x03 << 0); // 编码器模式 3 TIM4->CCER |= (1 << 0) | (1 << 4); // 启用 CH1 和 CH2 TIM4->CR1 |= (1 << 0); // 启用 TIM4 // TIM2 用于消毒定时 TIM2->PSC = 7199; // 72MHz / 7200 = 10kHz TIM2->ARR = 10000; // 10kHz / 10000 = 1Hz, 1秒中断 TIM2->DIER |= (1 << 0); // 启用更新中断 TIM2->CR1 |= (1 << 0); // 启用 TIM2 } // NVIC 初始化 void NVIC_Init(void) { // 启用 TIM2 中断 NVIC_ISER0 |= (1 << 28); // TIM2 中断号 28 // 配置 EXTI 用于人体传感器和防跌落传感器 AFIO_EXTICR1 &= ~(0xFF << 0); // 清除 EXTI0 和 EXTI1 配置 AFIO_EXTICR1 |= (0x01 << 0) | (0x01 << 4); // EXTI0 和 EXTI1 连接到 PB0 和 PB1 EXTI_RTSR |= (1 << 0) | (1 << 1); // 上升沿触发 EXTI_FTSR |= (1 << 0) | (1 << 1); // 下降沿触发 EXTI_IMR |= (1 << 0) | (1 << 1); // 启用 EXTI0 和 EXTI1 中断 NVIC_ISER0 |= (1 << 6) | (1 << 7); // EXTI0 中断号 6, EXTI1 中断号 7 } // 电机控制函数 void Motor_Control(uint8_t motor, int8_t speed) { if (motor == 0) { // 电机 A if (speed > 0) { GPIOA->BSRR = (1 << MOTOR_A_IN1_PIN); GPIOA->BRR = (1 << MOTOR_A_IN2_PIN); } else if (speed < 0) { GPIOA->BRR = (1 << MOTOR_A_IN1_PIN); GPIOA->BSRR = (1 << MOTOR_A_IN2_PIN); } else { GPIOA->BRR = (1 << MOTOR_A_IN1_PIN) | (1 << MOTOR_A_IN2_PIN); } TIM3->CCR1 = (speed < 0) ? -speed : speed; // PWM 占空比 } else if (motor == 1) { // 电机 B if (speed > 0) { GPIOA->BSRR = (1 << MOTOR_B_IN3_PIN); GPIOA->BRR = (1 << MOTOR_B_IN4_PIN); } else if (speed < 0) { GPIOA->BRR = (1 << MOTOR_B_IN3_PIN); GPIOA->BSRR = (1 << MOTOR_B_IN4_PIN); } else { GPIOA->BRR = (1 << MOTOR_B_IN3_PIN) | (1 << MOTOR_B_IN4_PIN); } TIM3->CCR2 = (speed < 0) ? -speed : speed; // PWM 占空比 } } // 紫外灯控制 void UV_Light_Control(uint8_t state) { if (state) { GPIOC->BSRR = (1 << UV_RELAY_PIN); // 开启紫外灯 } else { GPIOC->BRR = (1 << UV_RELAY_PIN); // 关闭紫外灯 } } // 读取人体传感器 uint8_t Read_Human_Sensor(void) { return (GPIOB->IDR & (1 << HUMAN_IR_PIN)) ? 1 : 0; } // 读取防跌落传感器 uint8_t Read_Drop_Sensor(void) { return (GPIOB->IDR & (1 << DROP_IR_PIN)) ? 1 : 0; } // 启动消毒定时器 void Disinfect_Timer_Start(uint32_t duration) { disinfect_time_remaining = duration; disinfect_active = 1; TIM2->CNT = 0; TIM2->CR1 |= (1 << 0); // 启用 TIM2 } // 停止消毒定时器 void Disinfect_Timer_Stop(void) { disinfect_active = 0; TIM2->CR1 &= ~(1 << 0); // 禁用 TIM2 UV_Light_Control(0); // 关闭紫外灯 // 返回充电坞逻辑可在此添加 } // 预设路径移动 void Move_Preset_Path(void) { // 示例路径: 前进 5 秒,左转 2 秒,前进 5 秒 Motor_Control(0, 80); // 电机 A 前进 Motor_Control(1, 80); // 电机 B 前进 for (volatile int i = 0; i < 5000000; i++); // 简单延时 Motor_Control(0, -80); // 电机 A 后退 Motor_Control(1, 80); // 电机 B 前进,左转 for (volatile int i = 0; i < 2000000; i++); Motor_Control(0, 80); Motor_Control(1, 80); for (volatile int i = 0; i < 5000000; i++); Motor_Control(0, 0); // 停止 Motor_Control(1, 0); } // TIM2 中断服务程序 void TIM2_IRQHandler(void) { if (TIM2->SR & (1 << 0)) { // 更新中断 TIM2->SR &= ~(1 << 0); // 清除中断标志 if (disinfect_active && disinfect_time_remaining > 0) { disinfect_time_remaining--; if (disinfect_time_remaining == 0) { Disinfect_Timer_Stop(); } } } } // EXTI0 中断服务程序 (人体传感器) void EXTI0_IRQHandler(void) { if (EXTI_PR & (1 << 0)) { EXTI_PR |= (1 << 0); // 清除中断标志 human_detected = Read_Human_Sensor(); if (human_detected) { UV_Light_Control(0); // 检测到人,立即关闭紫外灯 } } } // EXTI1 中断服务程序 (防跌落传感器) void EXTI1_IRQHandler(void) { if (EXTI_PR & (1 << 1)) { EXTI_PR |= (1 << 1); // 清除中断标志 drop_detected = Read_Drop_Sensor(); if (drop_detected) { Motor_Control(0, 0); // 停止电机 Motor_Control(1, 0); } } } // 主函数 int main(void) { SystemInit(); GPIO_Init(); TIM_Init(); NVIC_Init(); // 主循环 while (1) { // 检查传感器状态 human_detected = Read_Human_Sensor(); drop_detected = Read_Drop_Sensor(); // 无人且无跌落时启动消毒 if (!human_detected && !drop_detected && !disinfect_active) { Disinfect_Timer_Start(300); // 设置消毒时长为 300 秒(5 分钟) UV_Light_Control(1); Move_Preset_Path(); // 开始移动 } // 如果消毒激活,检查是否结束 if (disinfect_active && disinfect_time_remaining == 0) { Disinfect_Timer_Stop(); } // 防跌落处理 if (drop_detected) { Motor_Control(0, 0); Motor_Control(1, 0); } } } 项目核心代码#include "stm32f10x.h" // 引脚定义 #define MOTOR_IN1_PIN GPIO_Pin_0 #define MOTOR_IN2_PIN GPIO_Pin_1 #define MOTOR_IN3_PIN GPIO_Pin_2 #define MOTOR_IN4_PIN GPIO_Pin_3 #define MOTOR_PORT GPIOA #define UV_RELAY_PIN GPIO_Pin_4 #define UV_PORT GPIOA #define HUMAN_SENSOR_PIN GPIO_Pin_5 #define HUMAN_PORT GPIOA #define DROP_SENSOR_PIN GPIO_Pin_6 #define DROP_PORT GPIOA // 全局变量 volatile uint32_t uv_timer_count = 0; volatile uint8_t uv_enabled = 0; volatile uint8_t human_detected = 0; volatile uint8_t drop_detected = 0; // 函数声明 void SystemInit(void); void GPIO_Configuration(void); void TIM2_Configuration(void); void NVIC_Configuration(void); void Motor_Forward(void); void Motor_Backward(void); void Motor_Stop(void); void Motor_TurnLeft(void); void Motor_TurnRight(void); void UV_Light_On(void); void UV_Light_Off(void); uint8_t Read_Human_Sensor(void); uint8_t Read_Drop_Sensor(void); void Delay_ms(uint32_t ms); int main(void) { SystemInit(); GPIO_Configuration(); TIM2_Configuration(); NVIC_Configuration(); // 预设消毒时间(30分钟) uint32_t preset_time = 30 * 60; // 30分钟转换为秒 while(1) { // 读取传感器状态 human_detected = Read_Human_Sensor(); drop_detected = Read_Drop_Sensor(); // 安全条件检查:无人且无跌落风险 if(!human_detected && !drop_detected) { if(!uv_enabled) { // 开始消毒 UV_Light_On(); uv_enabled = 1; uv_timer_count = preset_time; TIM_Cmd(TIM2, ENABLE); // 启动定时器 } // 按照预设路径移动 Motor_Forward(); Delay_ms(2000); Motor_TurnRight(); Delay_ms(1000); Motor_Forward(); Delay_ms(2000); Motor_TurnLeft(); Delay_ms(1000); } else { // 安全条件不满足,停止消毒和移动 if(uv_enabled) { UV_Light_Off(); uv_enabled = 0; TIM_Cmd(TIM2, DISABLE); // 停止定时器 } Motor_Stop(); // 等待安全条件恢复 while(human_detected || drop_detected) { human_detected = Read_Human_Sensor(); drop_detected = Read_Drop_Sensor(); Delay_ms(100); } } // 检查消毒时间是否结束 if(uv_timer_count == 0 && uv_enabled) { UV_Light_Off(); uv_enabled = 0; TIM_Cmd(TIM2, DISABLE); // 返回充电坞 Motor_Backward(); Delay_ms(5000); // 假设返回充电坞需要5秒 Motor_Stop(); // 重置定时器 uv_timer_count = preset_time; } Delay_ms(100); } } void SystemInit(void) { // 启用时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN; RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; // 设置系统时钟 SystemCoreClockUpdate(); } void GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; // 电机控制引脚配置为推挽输出 GPIO_InitStructure.GPIO_Pin = MOTOR_IN1_PIN | MOTOR_IN2_PIN | MOTOR_IN3_PIN | MOTOR_IN4_PIN | UV_RELAY_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(MOTOR_PORT, &GPIO_InitStructure); // 传感器引脚配置为输入 GPIO_InitStructure.GPIO_Pin = HUMAN_SENSOR_PIN | DROP_SENSOR_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(HUMAN_PORT, &GPIO_InitStructure); } void TIM2_Configuration(void) { // 定时器2配置为1秒中断 TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_TimeBaseStructure.TIM_Period = 1000 - 1; // 1秒 TIM_TimeBaseStructure.TIM_Prescaler = 7200 - 1; // 72MHz/7200 = 10kHz TIM_TimeBaseStructure.TIM_ClockDivision = 0; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); TIM_Cmd(TIM2, DISABLE); } void NVIC_Configuration(void) { NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); } void TIM2_IRQHandler(void) { if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) { TIM_ClearITPendingBit(TIM2, TIM_IT_Update); if(uv_enabled && uv_timer_count > 0) { uv_timer_count--; } } } void Motor_Forward(void) { MOTOR_PORT->BSRR = MOTOR_IN1_PIN; // IN1高 MOTOR_PORT->BRR = MOTOR_IN2_PIN; // IN2低 MOTOR_PORT->BSRR = MOTOR_IN3_PIN; // IN3高 MOTOR_PORT->BRR = MOTOR_IN4_PIN; // IN4低 } void Motor_Backward(void) { MOTOR_PORT->BRR = MOTOR_IN1_PIN; // IN1低 MOTOR_PORT->BSRR = MOTOR_IN2_PIN; // IN2高 MOTOR_PORT->BRR = MOTOR_IN3_PIN; // IN3低 MOTOR_PORT->BSRR = MOTOR_IN4_PIN; // IN4高 } void Motor_Stop(void) { MOTOR_PORT->BRR = MOTOR_IN1_PIN | MOTOR_IN2_PIN | MOTOR_IN3_PIN | MOTOR_IN4_PIN; } void Motor_TurnLeft(void) { MOTOR_PORT->BRR = MOTOR_IN1_PIN; // IN1低 MOTOR_PORT->BRR = MOTOR_IN2_PIN; // IN2低 MOTOR_PORT->BSRR = MOTOR_IN3_PIN; // IN3高 MOTOR_PORT->BRR = MOTOR_IN4_PIN; // IN4低 } void Motor_TurnRight(void) { MOTOR_PORT->BSRR = MOTOR_IN1_PIN; // IN1高 MOTOR_PORT->BRR = MOTOR_IN2_PIN; // IN2低 MOTOR_PORT->BRR = MOTOR_IN3_PIN; // IN3低 MOTOR_PORT->BRR = MOTOR_IN4_PIN; // IN4低 } void UV_Light_On(void) { UV_PORT->BSRR = UV_RELAY_PIN; } void UV_Light_Off(void) { UV_PORT->BRR = UV_RELAY_PIN; } uint8_t Read_Human_Sensor(void) { return (HUMAN_PORT->IDR & HUMAN_SENSOR_PIN) ? 1 : 0; } uint8_t Read_Drop_Sensor(void) { return (DROP_PORT->IDR & DROP_SENSOR_PIN) ? 1 : 0; } void Delay_ms(uint32_t ms) { volatile uint32_t i, j; for(i = 0; i < ms; i++) for(j = 0; j < 7200; j++); } 总结实验室紫外灯智能消毒机器人是一种高效且安全的自动化设备,专为实验室内环境消毒设计。它通过预设路径自主移动,确保全面覆盖消毒区域,同时利用红外人体感应模块实时监测环境,仅在确认无人状态下才启动紫外灯消毒,有效避免对人体造成伤害。该机器人具备灵活的定时功能,用户可自定义消毒时长,任务完成后自动返回充电坞并关闭紫外灯,实现全流程自动化管理。此外,防跌落功能通过红外对管传感器实时检测边缘障碍,保障机器人在运行过程中的稳定与安全。硬件系统以STM32F103C8T6单片机为核心控制器,协调各模块工作。移动底盘采用带编码器的直流减速电机与L298N驱动板,确保精准路径跟踪;核心执行器包括紫外灯管和继电器模块,负责消毒操作;安全传感部分集成HC-SR501人体红外传感器和红外对管防跌落传感器,共同构建多层次防护机制。整体设计兼顾功能性与可靠性,适用于各类实验室场景。
  • [技术干货] 基于单片机的无线红外遥控密码锁
    项目开发背景随着社会对安全需求的日益增长,传统机械锁具在防护能力和使用便利性方面逐渐显现出局限性,例如钥匙易丢失、复制风险高以及缺乏灵活的访问控制机制。电子密码锁作为一种现代化解决方案,能够有效提升安全级别,并通过数字化管理减少物理钥匙的依赖。本项目基于单片机开发无线红外遥控密码锁,旨在融合本地与远程控制功能,满足用户对高效、智能安防系统的需求。在嵌入式系统领域,单片机以其低成本、高可靠性和易于编程的特点,成为智能设备的核心控制单元。本系统采用AT89C52单片机作为主控芯片,负责协调键盘输入、红外信号处理、密码验证及驱动执行模块,实现全流程自动化管理。通过集成4×4矩阵键盘和LCD1602显示屏,用户可直观地进行密码输入与状态监控,增强了人机交互的友好性。红外遥控技术的应用进一步扩展了系统的实用性,允许用户在无需直接接触设备的情况下进行远程操作。这种无线控制方式不仅提升了使用便利性,还适用于多种场景,例如在家庭或办公室环境中,用户可通过手持遥控器快速完成开锁,尤其适合行动不便或携带物品时的场景。为确保密码数据的持久性与安全性,系统采用AT24C02 EEPROM存储器,可在掉电情况下永久保存用户设置的密码,并支持灵活修改。这一设计避免了因电源中断导致的数据丢失问题,同时通过密码验证机制驱动继电器模拟开锁,或在错误输入时触发蜂鸣器报警,从而构建一个可靠的整体安防体系。该项目的开发体现了智能化安防技术的发展趋势,通过结合多种输入方式与稳定存储方案,不仅提升了锁具的防护性能,还为用户提供了便捷、可定制的使用体验。未来,这种系统可广泛应用于住宅、保险箱及办公场所,推动安全控制向更高效、人性化的方向演进。设计实现的功能(1) 通过4×4矩阵键盘输入密码,并在LCD1602液晶显示屏上实时显示输入状态(如数字或掩码符号)。(2) 支持通过红外遥控器进行远程密码输入与控制命令发送。(3) 允许用户修改密码,修改后的密码存储在AT24C02 EEPROM芯片中,确保掉电后数据不丢失。(4) 执行密码验证:若输入密码与存储密码匹配,则驱动5V继电器模拟开锁;若不匹配,则触发有源蜂鸣器发出报警声。项目硬件模块组成(1)主控芯片:AT89C52单片机。(2)输入模块:4×4矩阵键盘、红外遥控接收头。(3)显示模块:LCD1602液晶显示屏。(4)存储模块:AT24C02 EEPROM芯片。(5)执行与报警:5V继电器模块、有源蜂鸣器。设计意义这个基于单片机的无线红外遥控密码锁设计具有重要的实际意义。它提供了一种可靠的电子安全控制方案,通过密码验证机制来管理访问权限,有效防止未授权进入,适用于家庭、办公室等需要基本安防的场所,提升了日常生活的安全性。系统支持多种输入方式,包括本地4×4矩阵键盘和远程红外遥控,用户可以根据环境需求灵活选择操作模式,这大大增强了使用的便捷性和适应性,尤其适用于需要远程控制或隐蔽操作的场景。密码可修改并存储在EEPROM存储器中,确保数据在断电后不丢失,这提高了系统的持久性和可靠性,用户能够根据需要随时更新密码,从而维护长期的安全保障。验证机制中,正确密码驱动继电器模拟开锁,错误则触发蜂鸣器报警,实现了快速响应和警示功能,在实际应用中能及时反馈操作状态,避免潜在风险。设计思路该系统以AT89C52单片机作为核心控制器,负责协调各个硬件模块实现密码锁功能。用户可以通过4×4矩阵键盘进行本地密码输入,同时支持红外遥控器进行远程输入与控制,单片机通过扫描键盘行列和解析红外信号来获取用户输入数据。LCD1602液晶显示屏用于实时显示输入状态,例如密码位数、输入提示或错误信息,确保用户操作过程清晰可见。显示内容由单片机控制更新,初始界面可设置欢迎语或操作指南。密码存储采用AT24C02 EEPROM芯片,通过I2C协议与单片机通信,实现密码的读写操作。用户可修改密码,新密码经验证后存入EEPROM,确保掉电后数据不丢失,增强了系统的可靠性。密码验证环节中,单片机会比较输入密码与存储密码,若匹配则驱动5V继电器模块模拟开锁动作;若不匹配则触发有源蜂鸣器发出报警声,以提示用户输入错误。整个流程兼顾安全性与实用性,无需额外功能扩展。框架图系统框架图:+-----------------------------+ | AT89C52单片机 | | (主控芯片) | +-----------------------------+ | | (数据/控制总线) | +--------+--------+--------+--------+--------+ | | | | | | v v v v v v +-------+ +-------+ +-------+ +-------+ +-------+ +-------+ | 4×4 | |红外 | |LCD1602| |AT24C02| |5V | |有源 | |矩阵键盘| |遥控 | |显示屏 | |EEPROM | |继电器 | |蜂鸣器 | | | |接收头 | | | | | |模块 | | | +-------+ +-------+ +-------+ +-------+ +-------+ +-------+ (输入) (输入) (显示) (存储) (执行) (报警)系统总体设计该系统基于AT89C52单片机作为核心控制器,实现一个集成无线红外遥控和本地键盘输入的密码锁系统。系统通过协调多个硬件模块完成密码输入、验证、显示和执行功能,确保安全可靠的操作。输入模块包括4×4矩阵键盘和红外遥控接收头,用户可通过键盘直接输入密码或使用红外遥控器进行远程操作。LCD1602液晶显示屏实时显示密码输入状态和系统提示信息,为用户提供直观的交互反馈。密码数据存储在AT24C02 EEPROM芯片中,支持用户修改密码并保证掉电后数据不丢失。系统在验证密码时,若输入正确则驱动5V继电器模块模拟开锁动作;若错误则触发有源蜂鸣器发出报警信号,以增强安全性。整个系统由AT89C52单片机统一管理,负责处理输入信号、控制显示内容、访问存储数据以及驱动执行模块,确保各部件协同工作,实现高效稳定的密码锁功能。系统功能总结功能描述硬件模块本地密码输入与显示用户通过4×4矩阵键盘输入密码,LCD1602液晶显示屏实时显示输入状态。4×4矩阵键盘、LCD1602远程密码输入与控制支持通过红外遥控器进行远程密码输入与系统控制。红外遥控接收头密码存储与修改密码可修改,并存储在AT24C02 EEPROM芯片中,确保掉电后数据不丢失。AT24C02 EEPROM芯片密码验证与执行密码验证正确则驱动5V继电器模拟开锁;错误则触发有源蜂鸣器报警。5V继电器模块、有源蜂鸣器系统核心控制AT89C52单片机作为主控芯片,协调输入、显示、存储和执行模块的整体运作。AT89C52单片机设计的各个功能模块描述主控芯片模块采用AT89C52单片机作为系统核心控制器,负责协调整个密码锁的运行。它处理来自输入模块的密码数据,执行密码验证逻辑,控制显示模块的输出信息,管理存储模块的读写操作,并根据验证结果驱动执行与报警模块的动作。输入模块包括4×4矩阵键盘和红外遥控接收头,用于接收用户密码输入。矩阵键盘提供本地密码输入方式,用户可通过按键输入数字密码;红外遥控接收头支持远程控制,允许用户通过红外遥控器进行密码输入和系统操作,实现无线遥控功能。显示模块使用LCD1602液晶显示屏,实时显示密码输入状态和系统提示信息。例如,在用户输入密码时,屏幕会显示输入的数字或星号以隐藏真实密码,并在验证过程中给出正确或错误的反馈提示,确保用户操作直观明了。存储模块基于AT24C02 EEPROM芯片,用于存储用户设置的密码数据。该模块支持密码修改功能,用户可通过输入新密码并保存到EEPROM中,且存储的数据在系统断电后不会丢失,保证了密码的持久性和安全性。执行与报警模块由5V继电器模块和有源蜂鸣器组成。当密码验证正确时,继电器被驱动以模拟开锁动作;如果密码输入错误,系统会触发有源蜂鸣器发出报警声,提醒用户并增强安全防护。上位机代码设计#include <QApplication> #include <QMainWindow> #include <QVBoxLayout> #include <QHBoxLayout> #include <QPushButton> #include <QLineEdit> #include <QLabel> #include <QTextEdit> #include <QSerialPort> #include <QSerialPortInfo> #include <QMessageBox> #include <QTimer> #include <QComboBox> class RemoteLockController : public QMainWindow { Q_OBJECT public: RemoteLockController(QWidget *parent = nullptr) : QMainWindow(parent) { setupUI(); setupSerialPort(); setupConnections(); } private slots: void connectToDevice() { if(serialPort->isOpen()) { serialPort->close(); } serialPort->setPortName(portComboBox->currentText()); serialPort->setBaudRate(QSerialPort::Baud9600); serialPort->setDataBits(QSerialPort::Data8); serialPort->setParity(QSerialPort::NoParity); serialPort->setStopBits(QSerialPort::OneStop); serialPort->setFlowControl(QSerialPort::NoFlowControl); if(serialPort->open(QIODevice::ReadWrite)) { statusLabel->setText("已连接到: " + portComboBox->currentText()); logTextEdit->append("系统: 连接成功"); } else { statusLabel->setText("连接失败"); logTextEdit->append("系统: 连接失败"); } } void sendPassword() { QString password = passwordLineEdit->text(); if(password.isEmpty()) { QMessageBox::warning(this, "警告", "密码不能为空"); return; } if(serialPort->isOpen()) { QString command = "PASS:" + password + "\n"; serialPort->write(command.toUtf8()); logTextEdit->append("发送: " + command.trimmed()); } else { QMessageBox::warning(this, "警告", "请先连接设备"); } } void changePassword() { QString newPassword = newPasswordLineEdit->text(); if(newPassword.isEmpty()) { QMessageBox::warning(this, "警告", "新密码不能为空"); return; } if(serialPort->isOpen()) { QString command = "CHANGE:" + newPassword + "\n"; serialPort->write(command.toUtf8()); logTextEdit->append("发送: " + command.trimmed()); } else { QMessageBox::warning(this, "警告", "请先连接设备"); } } void unlockDoor() { if(serialPort->isOpen()) { QString command = "UNLOCK\n"; serialPort->write(command.toUtf8()); logTextEdit->append("发送: " + command.trimmed()); } else { QMessageBox::warning(this, "警告", "请先连接设备"); } } void readData() { QByteArray data = serialPort->readAll(); if(!data.isEmpty()) { QString response = QString::fromUtf8(data).trimmed(); logTextEdit->append("接收: " + response); if(response.contains("SUCCESS")) { QMessageBox::information(this, "成功", "操作成功"); } else if(response.contains("ERROR")) { QMessageBox::warning(this, "错误", "操作失败"); } else if(response.contains("ALARM")) { QMessageBox::critical(this, "警报", "密码错误触发警报"); } } } void refreshPorts() { portComboBox->clear(); QList<QSerialPortInfo> ports = QSerialPortInfo::availablePorts(); for(const QSerialPortInfo &port : ports) { portComboBox->addItem(port.portName()); } } private: void setupUI() { QWidget *centralWidget = new QWidget(this); setCentralWidget(centralWidget); QVBoxLayout *mainLayout = new QVBoxLayout(centralWidget); // 连接设置区域 QHBoxLayout *connectionLayout = new QHBoxLayout(); portComboBox = new QComboBox(); refreshButton = new QPushButton("刷新端口"); connectButton = new QPushButton("连接设备"); statusLabel = new QLabel("未连接"); connectionLayout->addWidget(new QLabel("串口:")); connectionLayout->addWidget(portComboBox); connectionLayout->addWidget(refreshButton); connectionLayout->addWidget(connectButton); connectionLayout->addWidget(statusLabel); connectionLayout->addStretch(); // 密码操作区域 QHBoxLayout *passwordLayout = new QHBoxLayout(); passwordLineEdit = new QLineEdit(); passwordLineEdit->setPlaceholderText("输入密码"); passwordLineEdit->setEchoMode(QLineEdit::Password); sendPasswordButton = new QPushButton("发送密码"); passwordLayout->addWidget(new QLabel("密码验证:")); passwordLayout->addWidget(passwordLineEdit); passwordLayout->addWidget(sendPasswordButton); // 修改密码区域 QHBoxLayout *changePasswordLayout = new QHBoxLayout(); newPasswordLineEdit = new QLineEdit(); newPasswordLineEdit->setPlaceholderText("输入新密码"); newPasswordLineEdit->setEchoMode(QLineEdit::Password); changePasswordButton = new QPushButton("修改密码"); changePasswordLayout->addWidget(new QLabel("修改密码:")); changePasswordLayout->addWidget(newPasswordLineEdit); changePasswordLayout->addWidget(changePasswordButton); // 控制区域 QHBoxLayout *controlLayout = new QHBoxLayout(); unlockButton = new QPushButton("远程开锁"); controlLayout->addWidget(unlockButton); controlLayout->addStretch(); // 日志区域 logTextEdit = new QTextEdit(); logTextEdit->setReadOnly(true); mainLayout->addLayout(connectionLayout); mainLayout->addLayout(passwordLayout); mainLayout->addLayout(changePasswordLayout); mainLayout->addLayout(controlLayout); mainLayout->addWidget(new QLabel("通信日志:")); mainLayout->addWidget(logTextEdit); setWindowTitle("无线红外遥控密码锁上位机"); resize(600, 400); refreshPorts(); } void setupSerialPort() { serialPort = new QSerialPort(this); } void setupConnections() { connect(connectButton, &QPushButton::clicked, this, &RemoteLockController::connectToDevice); connect(sendPasswordButton, &QPushButton::clicked, this, &RemoteLockController::sendPassword); connect(changePasswordButton, &QPushButton::clicked, this, &RemoteLockController::changePassword); connect(unlockButton, &QPushButton::clicked, this, &RemoteLockController::unlockDoor); connect(refreshButton, &QPushButton::clicked, this, &RemoteLockController::refreshPorts); connect(serialPort, &QSerialPort::readyRead, this, &RemoteLockController::readData); } QSerialPort *serialPort; QComboBox *portComboBox; QPushButton *refreshButton; QPushButton *connectButton; QLabel *statusLabel; QLineEdit *passwordLineEdit; QPushButton *sendPasswordButton; QLineEdit *newPasswordLineEdit; QPushButton *changePasswordButton; QPushButton *unlockButton; QTextEdit *logTextEdit; }; int main(int argc, char *argv[]) { QApplication app(argc, argv); RemoteLockController controller; controller.show(); return app.exec(); } #include "main.moc" 项目配置文件 (CMakeLists.txt):cmake_minimum_required(VERSION 3.16) project(RemoteLockController) set(CMAKE_CXX_STANDARD 17) set(CMAKE_CXX_STANDARD_REQUIRED ON) find_package(Qt6 REQUIRED COMPONENTS Core Widgets SerialPort) qt_standard_project_setup() qt_add_executable(RemoteLockController main.cpp) target_link_libraries(RemoteLockController Qt6::Core Qt6::Widgets Qt6::SerialPort) 项目配置文件替代方案 (qmake .pro文件):QT += core widgets serialport CONFIG += c++17 SOURCES += \ main.cpp HEADERS += # Default rules for deployment. qnx: target.path = /tmp/$${TARGET}/bin else: unix:!android: target.path = /opt/$${TARGET}/bin !isEmpty(target.path): INSTALLS += target这个上位机程序提供了完整的密码锁控制功能,包括:串口连接管理密码验证密码修改远程开锁实时通信日志状态反馈显示程序使用Qt框架开发,支持跨平台运行,通过串口与下位机AT89C52单片机进行通信。模块代码设计#include "stm32f10x.h" // LCD1602引脚定义 #define LCD_RS_PIN GPIO_Pin_0 #define LCD_RW_PIN GPIO_Pin_1 #define LCD_EN_PIN GPIO_Pin_2 #define LCD_D4_PIN GPIO_Pin_3 #define LCD_D5_PIN GPIO_Pin_4 #define LCD_D6_PIN GPIO_Pin_5 #define LCD_D7_PIN GPIO_Pin_6 #define LCD_PORT GPIOA // 矩阵键盘引脚定义 #define KEY_ROW1 GPIO_Pin_0 #define KEY_ROW2 GPIO_Pin_1 #define KEY_ROW3 GPIO_Pin_2 #define KEY_ROW4 GPIO_Pin_3 #define KEY_COL1 GPIO_Pin_4 #define KEY_COL2 GPIO_Pin_5 #define KEY_COL3 GPIO_Pin_6 #define KEY_COL4 GPIO_Pin_7 #define KEY_ROW_PORT GPIOB #define KEY_COL_PORT GPIOB // 红外接收引脚 #define IR_PIN GPIO_Pin_8 #define IR_PORT GPIOA // 继电器和蜂鸣器引脚 #define RELAY_PIN GPIO_Pin_13 #define BUZZER_PIN GPIO_Pin_14 #define OUTPUT_PORT GPIOC // EEPROM地址定义 #define EEPROM_ADDR 0xA0 #define PASSWORD_ADDR 0x00 // 全局变量 uint8_t input_password[6] = {0}; uint8_t stored_password[6] = {0}; uint8_t input_index = 0; uint32_t ir_value = 0; // 延时函数 void delay_ms(uint16_t ms) { uint32_t i; for(i = 0; i < ms * 8000; i++); } // LCD1602写命令 void lcd_write_cmd(uint8_t cmd) { GPIO_ResetBits(LCD_PORT, LCD_RS_PIN); // RS=0 GPIO_ResetBits(LCD_PORT, LCD_RW_PIN); // RW=0 // 高4位 GPIO_WriteBit(LCD_PORT, LCD_D4_PIN, (cmd >> 4) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D5_PIN, (cmd >> 5) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D6_PIN, (cmd >> 6) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D7_PIN, (cmd >> 7) & 0x01 ? Bit_SET : Bit_RESET); GPIO_SetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); GPIO_ResetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); // 低4位 GPIO_WriteBit(LCD_PORT, LCD_D4_PIN, (cmd >> 0) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D5_PIN, (cmd >> 1) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D6_PIN, (cmd >> 2) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D7_PIN, (cmd >> 3) & 0x01 ? Bit_SET : Bit_RESET); GPIO_SetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); GPIO_ResetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); } // LCD1602写数据 void lcd_write_data(uint8_t dat) { GPIO_SetBits(LCD_PORT, LCD_RS_PIN); // RS=1 GPIO_ResetBits(LCD_PORT, LCD_RW_PIN); // RW=0 // 高4位 GPIO_WriteBit(LCD_PORT, LCD_D4_PIN, (dat >> 4) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D5_PIN, (dat >> 5) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D6_PIN, (dat >> 6) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D7_PIN, (dat >> 7) & 0x01 ? Bit_SET : Bit_RESET); GPIO_SetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); GPIO_ResetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); // 低4位 GPIO_WriteBit(LCD_PORT, LCD_D4_PIN, (dat >> 0) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D5_PIN, (dat >> 1) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D6_PIN, (dat >> 2) & 0x01 ? Bit_SET : Bit_RESET); GPIO_WriteBit(LCD_PORT, LCD_D7_PIN, (dat >> 3) & 0x01 ? Bit_SET : Bit_RESET); GPIO_SetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); GPIO_ResetBits(LCD_PORT, LCD_EN_PIN); delay_ms(2); } // LCD1602初始化 void lcd_init(void) { delay_ms(15); lcd_write_cmd(0x33); delay_ms(5); lcd_write_cmd(0x32); delay_ms(5); lcd_write_cmd(0x28); // 4位模式,2行显示 lcd_write_cmd(0x0C); // 显示开,光标关 lcd_write_cmd(0x06); // 写入数据后光标右移 lcd_write_cmd(0x01); // 清屏 delay_ms(2); } // LCD显示字符串 void lcd_display_string(uint8_t x, uint8_t y, char *str) { uint8_t addr; if(y == 0) addr = 0x80 + x; else addr = 0xC0 + x; lcd_write_cmd(addr); while(*str) { lcd_write_data(*str++); } } // 矩阵键盘扫描 uint8_t key_scan(void) { uint8_t row, col, key_val = 0; GPIO_InitTypeDef GPIO_InitStructure; // 设置列为输出,行为输入 GPIO_InitStructure.GPIO_Pin = KEY_COL1 | KEY_COL2 | KEY_COL3 | KEY_COL4; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(KEY_COL_PORT, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = KEY_ROW1 | KEY_ROW2 | KEY_ROW3 | KEY_ROW4; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; GPIO_Init(KEY_ROW_PORT, &GPIO_InitStructure); // 列扫描 for(col = 0; col < 4; col++) { GPIO_SetBits(KEY_COL_PORT, KEY_COL1 | KEY_COL2 | KEY_COL3 | KEY_COL4); GPIO_ResetBits(KEY_COL_PORT, (KEY_COL1 << col)); delay_ms(10); // 行检测 if(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW1) == 0) { key_val = 1 + col; while(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW1) == 0); break; } if(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW2) == 0) { key_val = 5 + col; while(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW2) == 0); break; } if(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW3) == 0) { key_val = 9 + col; while(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW3) == 0); break; } if(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW4) == 0) { key_val = 13 + col; while(GPIO_ReadInputDataBit(KEY_ROW_PORT, KEY_ROW4) == 0); break; } } return key_val; } // I2C初始化 void i2c_init(void) { RCC->APB2ENR |= 1 << 3; // 使能GPIOB时钟 RCC->APB1ENR |= 1 << 21; // 使能I2C1时钟 // PB6-SCL, PB7-SDA GPIOB->CRL &= 0x00FFFFFF; GPIOB->CRL |= 0xFF000000; // 推挽输出,50MHz GPIOB->ODR |= 0xC0; // 输出高电平 I2C1->CR1 |= 1 << 15; // 软件复位 I2C1->CR1 &= ~(1 << 15); I2C1->CR2 = 36; // 36MHz I2C1->CCR = 180; // 100kHz I2C1->TRISE = 37; // 最大上升时间 I2C1->CR1 |= 1 << 0; // 使能I2C } // I2C起始信号 void i2c_start(void) { I2C1->CR1 |= 1 << 8; // 产生起始条件 while(!(I2C1->SR1 & 0x0001)); // 等待起始条件完成 } // I2C停止信号 void i2c_stop(void) { I2C1->CR1 |= 1 << 9; // 产生停止条件 while(I2C1->SR2 & 0x0002); // 等待总线空闲 } // I2C发送一个字节 void i2c_send_byte(uint8_t dat) { I2C1->DR = dat; while(!(I2C1->SR1 & 0x0002)); // 等待数据传输完成 } // I2C接收一个字节 uint8_t i2c_read_byte(void) { while(!(I2C1->SR1 & 0x0004)); // 等待接收完成 return I2C1->DR; } // EEPROM写一个字节 void eeprom_write_byte(uint8_t addr, uint8_t dat) { i2c_start(); i2c_send_byte(EEPROM_ADDR); i2c_send_byte(addr); i2c_send_byte(dat); i2c_stop(); delay_ms(10); // 等待写入完成 } // EEPROM读一个字节 uint8_t eeprom_read_byte(uint8_t addr) { uint8_t dat; i2c_start(); i2c_send_byte(EEPROM_ADDR); i2c_send_byte(addr); i2c_start(); i2c_send_byte(EEPROM_ADDR | 0x01); dat = i2c_read_byte(); i2c_stop(); return dat; } // 保存密码到EEPROM void save_password(uint8_t *pwd) { uint8_t i; for(i = 0; i < 6; i++) { eeprom_write_byte(PASSWORD_ADDR + i, pwd[i]); } } // 从EEPROM读取密码 void read_password(uint8_t *pwd) { uint8_t i; for(i = 0; i < 6; i++) { pwd[i] = eeprom_read_byte(PASSWORD_ADDR + i); } } // 红外接收初始化 void ir_init(void) { RCC->APB2ENR |= 1 << 2; // 使能GPIOA时钟 RCC->APB1ENR |= 1 << 0; // 使能TIM2时钟 // PA8输入 GPIOA->CRH &= 0xFFFFFFF0; GPIOA->CRH |= 0x00000004; // 浮空输入 // TIM2初始化 TIM2->PSC = 71; // 1MHz TIM2->ARR = 10000; // 10ms TIM2->CCMR1 |= 1 << 0; // CC1为输入,映射到TI1 TIM2->CCER |= 1 << 0; // CC1使能 TIM2->DIER |= 1 << 0; // 更新中断使能 TIM2->DIER |= 1 << 1; // CC1中断使能 TIM2->CR1 |= 1 << 0; // 使能TIM2 NVIC->ISER[0] |= 1 << 28; // 使能TIM2中断 } // 红外解码 void ir_decode(void) { uint32_t i, j; uint32_t data = 0; if(GPIO_ReadInputDataBit(IR_PORT, IR_PIN) == 0) { delay_ms(1); if(GPIO_ReadInputDataBit(IR_PORT, IR_PIN) == 0) { // 确认引导码 while(GPIO_ReadInputDataBit(IR_PORT, IR_PIN) == 0); for(i = 0; i < 4; i++) { for(j = 0; j < 8; j++) { while(GPIO_ReadInputDataBit(IR_PORT, IR_PIN) == 0); delay_ms(1); if(GPIO_ReadInputDataBit(IR_PORT, IR_PIN)) { data |= (1 << (i * 8 + j)); while(GPIO_ReadInputDataBit(IR_PORT, IR_PIN)); } } } ir_value = data; } } } // 继电器控制 void relay_control(uint8_t state) { if(state) { GPIO_SetBits(OUTPUT_PORT, RELAY_PIN); } else { GPIO_ResetBits(OUTPUT_PORT, RELAY_PIN); } } // 蜂鸣器控制 void buzzer_control(uint8_t state) { if(state) { GPIO_SetBits(OUTPUT_PORT, BUZZER_PIN); } else { GPIO_ResetBits(OUTPUT_PORT, BUZZER_PIN); } } // 密码验证 uint8_t verify_password(void) { uint8_t i; for(i = 0; i < 6; i++) { if(input_password[i] != stored_password[i]) { return 0; } } return 1; } // 系统初始化 void system_init(void) { RCC->APB2ENR |= 1 << 2; // 使能GPIOA时钟 RCC->APB2ENR |= 1 << 3; // 使能GPIOB时钟 RCC->APB2ENR |= 1 << 4; // 使能GPIOC时钟 // LCD初始化 GPIOA->CRL = 0x33333333; // PA0-PA7推挽输出 // 输出控制初始化 GPIOC->CRH &= 0x00FFFFFF; GPIOC->CRH |= 0x33000000; // PC13,PC14推挽输出 lcd_init(); i2c_init(); ir_init(); // 读取存储的密码 read_password(stored_password); } int main(void) { uint8_t key, i; system_init(); lcd_display_string(0, 0, "Password Lock"); lcd_display_string(0, 1, "Input:******"); while(1) { // 键盘输入 key = key_scan(); if(key != 0 && input_index < 6) { if(key <= 9) { // 数字键 input_password[input_index] = key; lcd_write_cmd(0xC0 + 6 + input_index); lcd_write_data('*'); input_index++; } else if(key == 14) { // 删除键 if(input_index > 0) { input_index--; input_password[input_index] = 0; lcd_write_cmd(0xC0 + 6 + input_index); lcd_write_data(' '); } } else if(key == 15) { // 确认键 if(input_index == 6) { if(verify_password()) { relay_control(1); lcd_display_string(0, 1, "Open Success! "); delay_ms(2000); relay_control(0); } else { buzzer_control(1); lcd_display_string(0, 1, "Error! Try Again"); delay_ms(2000); buzzer_control(0); } // 清空输入 for(i = 0; i < 6; i++) { input_password[i] = 0; } input_index = 0; lcd_display_string(0, 1, "Input:******"); } } else if(key == 16) { // 修改密码键 if(input_index == 6) { save_password(input_password); read_password(stored_password); lcd_display_string(0, 1, "Pwd Changed! "); delay_ms(2000); for(i = 0; i < 6; i++) { input_password[i] = 0; } input_index = 0; lcd_display_string(0, 1, "Input:******"); } } } // 红外输入处理 ir_decode(); if(ir_value != 0) { // 根据红外码值处理 // 这里需要根据实际遥控器编码进行映射 // 简化处理:直接当作数字输入 if(ir_value >= 0x00 && ir_value <= 0x09 && input_index < 6) { input_password[input_index] = ir_value; lcd_write_cmd(0xC0 + 6 + input_index); lcd_write_data('*'); input_index++; } ir_value = 0; } delay_ms(100); } } // TIM2中断服务函数 void TIM2_IRQHandler(void) { if(TIM2->SR & 0x0001) { // 更新中断 TIM2->SR &= ~0x0001; } if(TIM2->SR & 0x0002) { // CC1中断 TIM2->SR &= ~0x0002; ir_decode(); } } 项目核心代码#include "stm32f10x.h" // 密码长度定义 #define PASSWORD_LENGTH 6 // 系统状态枚举 typedef enum { MODE_NORMAL, MODE_SETTING_VERIFY_OLD, MODE_SETTING_INPUT_NEW, MODE_SETTING_CONFIRM_NEW } SystemMode; // 全局变量 uint8_t storedPassword[PASSWORD_LENGTH] = {1, 2, 3, 4, 5, 6}; // 默认密码 uint8_t inputBuffer[PASSWORD_LENGTH]; uint8_t inputIndex = 0; uint8_t newPassword[PASSWORD_LENGTH]; SystemMode currentMode = MODE_NORMAL; // 外设初始化函数声明 void GPIO_Init(void); void LCD_Init(void); void Keyboard_Init(void); void EEPROM_Init(void); void Infrared_Init(void); void Timer_Init(void); // 功能模块函数声明 uint8_t Keyboard_Scan(void); uint8_t Infrared_GetKey(void); void LCD_Clear(void); void LCD_DisplayString(char *str); void LCD_DisplayChar(char c); void EEPROM_Write(uint16_t addr, uint8_t data); uint8_t EEPROM_Read(uint16_t addr); void Relay_Control(uint8_t state); void Buzzer_Control(uint8_t state); void Delay_ms(uint32_t ms); // 密码管理函数 void ReadPasswordFromEEPROM(void); void WritePasswordToEEPROM(void); uint8_t VerifyPassword(uint8_t *input, uint8_t *stored); void ProcessKey(uint8_t key); void CheckPassword(void); void EnterSettingMode(void); void ExitSettingMode(void); void ResetInputBuffer(void); int main(void) { // 系统初始化 SystemInit(); GPIO_Init(); LCD_Init(); Keyboard_Init(); EEPROM_Init(); Infrared_Init(); Timer_Init(); // 从EEPROM读取存储的密码 ReadPasswordFromEEPROM(); // 初始显示 LCD_Clear(); LCD_DisplayString("Enter Password:"); LCD_DisplayString("\n* for Settings"); while(1) { uint8_t key; // 扫描矩阵键盘 key = Keyboard_Scan(); if(key != 0xFF) { ProcessKey(key); Delay_ms(200); // 去抖动延时 } // 处理红外遥控 key = Infrared_GetKey(); if(key != 0xFF) { ProcessKey(key); } // 根据当前模式更新显示 switch(currentMode) { case MODE_NORMAL: if(inputIndex == 0) { LCD_Clear(); LCD_DisplayString("Enter Password:"); } break; case MODE_SETTING_VERIFY_OLD: LCD_Clear(); LCD_DisplayString("Verify Old Pwd:"); break; case MODE_SETTING_INPUT_NEW: LCD_Clear(); LCD_DisplayString("Enter New Pwd:"); break; case MODE_SETTING_CONFIRM_NEW: LCD_Clear(); LCD_DisplayString("Confirm New Pwd:"); break; } } } void GPIO_Init(void) { // 使能GPIO时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN | RCC_APB2ENR_IOPCEN | RCC_APB2ENR_IOPDEN; // 配置继电器控制引脚(PC0)为推挽输出 GPIOC->CRL &= ~GPIO_CRL_MODE0; GPIOC->CRL |= GPIO_CRL_MODE0_0; GPIOC->CRL &= ~GPIO_CRL_CNF0; // 配置蜂鸣器控制引脚(PC1)为推挽输出 GPIOC->CRL &= ~GPIO_CRL_MODE1; GPIOC->CRL |= GPIO_CRL_MODE1_0; GPIOC->CRL &= ~GPIO_CRL_CNF1; // 初始状态关闭 Relay_Control(0); Buzzer_Control(0); } void Timer_Init(void) { // 基本定时器配置 RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; TIM2->PSC = 7200 - 1; // 10kHz TIM2->ARR = 10000 - 1; // 1s TIM2->CR1 |= TIM_CR1_ARPE; TIM2->CR1 |= TIM_CR1_CEN; } void ProcessKey(uint8_t key) { switch(key) { case 0x0A: // '*'键 - 进入设置模式 if(currentMode == MODE_NORMAL) { EnterSettingMode(); } else { // 设置模式下按'*'取消 currentMode = MODE_NORMAL; ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Settings Canceled"); Delay_ms(1000); LCD_Clear(); LCD_DisplayString("Enter Password:"); } break; case 0x0B: // '#'键 - 确认输入 if(currentMode == MODE_NORMAL) { CheckPassword(); } else if(currentMode == MODE_SETTING_VERIFY_OLD) { if(VerifyPassword(inputBuffer, storedPassword)) { currentMode = MODE_SETTING_INPUT_NEW; ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Enter New Pwd:"); } else { Buzzer_Control(1); LCD_Clear(); LCD_DisplayString("Wrong Password!"); Delay_ms(1000); Buzzer_Control(0); currentMode = MODE_NORMAL; ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Enter Password:"); } } else if(currentMode == MODE_SETTING_INPUT_NEW) { // 保存新密码到临时缓冲区 for(int i = 0; i < PASSWORD_LENGTH; i++) { newPassword[i] = inputBuffer[i]; } currentMode = MODE_SETTING_CONFIRM_NEW; ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Confirm New Pwd:"); } else if(currentMode == MODE_SETTING_CONFIRM_NEW) { // 验证两次输入的新密码是否一致 uint8_t match = 1; for(int i = 0; i < PASSWORD_LENGTH; i++) { if(inputBuffer[i] != newPassword[i]) { match = 0; break; } } if(match) { // 保存新密码 for(int i = 0; i < PASSWORD_LENGTH; i++) { storedPassword[i] = newPassword[i]; } WritePasswordToEEPROM(); LCD_Clear(); LCD_DisplayString("Password Changed!"); Delay_ms(1000); ExitSettingMode(); } else { Buzzer_Control(1); LCD_Clear(); LCD_DisplayString("Password Mismatch!"); Delay_ms(1000); Buzzer_Control(0); currentMode = MODE_SETTING_INPUT_NEW; ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Enter New Pwd:"); } } break; default: // 数字键0-9 if(key >= 0 && key <= 9) { if(inputIndex < PASSWORD_LENGTH) { inputBuffer[inputIndex++] = key; LCD_DisplayChar('*'); // 自动验证(可选) if(inputIndex == PASSWORD_LENGTH && currentMode == MODE_NORMAL) { CheckPassword(); } } } break; } } void CheckPassword(void) { if(VerifyPassword(inputBuffer, storedPassword)) { Relay_Control(1); LCD_Clear(); LCD_DisplayString("Access Granted!"); Delay_ms(3000); // 保持开锁状态3秒 Relay_Control(0); } else { Buzzer_Control(1); LCD_Clear(); LCD_DisplayString("Access Denied!"); Delay_ms(1000); Buzzer_Control(0); } ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Enter Password:"); } void EnterSettingMode(void) { currentMode = MODE_SETTING_VERIFY_OLD; ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Verify Old Pwd:"); } void ExitSettingMode(void) { currentMode = MODE_NORMAL; ResetInputBuffer(); LCD_Clear(); LCD_DisplayString("Enter Password:"); } void ResetInputBuffer(void) { for(int i = 0; i < PASSWORD_LENGTH; i++) { inputBuffer[i] = 0; } inputIndex = 0; } uint8_t VerifyPassword(uint8_t *input, uint8_t *stored) { for(int i = 0; i < PASSWORD_LENGTH; i++) { if(input[i] != stored[i]) { return 0; } } return 1; } void ReadPasswordFromEEPROM(void) { for(int i = 0; i < PASSWORD_LENGTH; i++) { storedPassword[i] = EEPROM_Read(i); // 如果EEPROM中无数据,使用默认密码 if(storedPassword[i] > 9) { storedPassword[i] = i + 1; } } } void WritePasswordToEEPROM(void) { for(int i = 0; i < PASSWORD_LENGTH; i++) { EEPROM_Write(i, storedPassword[i]); } } void Relay_Control(uint8_t state) { if(state) { GPIOC->BSRR = GPIO_BSRR_BS0; // 置位PC0 } else { GPIOC->BSRR = GPIO_BSRR_BR0; // 复位PC0 } } void Buzzer_Control(uint8_t state) { if(state) { GPIOC->BSRR = GPIO_BSRR_BS1; // 置位PC1 } else { GPIOC->BSRR = GPIO_BSRR_BR1; // 复位PC1 } } void Delay_ms(uint32_t ms) { for(uint32_t i = 0; i < ms; i++) { for(uint32_t j = 0; j < 7200; j++) { __NOP(); } } } // 以下函数假设已在其他模块中实现 // 这里只提供声明,实际开发时需要具体实现 /* void LCD_Init(void) { } void Keyboard_Init(void) { } void EEPROM_Init(void) { } void Infrared_Init(void) { } uint8_t Keyboard_Scan(void) { return 0xFF; } uint8_t Infrared_GetKey(void) { return 0xFF; } void LCD_Clear(void) { } void LCD_DisplayString(char *str) { } void LCD_DisplayChar(char c) { } void EEPROM_Write(uint16_t addr, uint8_t data) { } uint8_t EEPROM_Read(uint16_t addr) { return 0; } */ 总结该系统以AT89C52单片机为核心,成功实现了无线红外遥控密码锁的设计。通过集成4×4矩阵键盘和红外遥控接收头,用户可灵活选择本地或远程方式输入密码,同时LCD1602液晶屏实时显示操作状态,提升了交互便利性。密码管理功能支持用户修改并存储于AT24C02 EEPROM芯片中,确保数据在掉电后不丢失,增强了系统的可靠性和长期使用价值。执行模块中,密码验证正确后驱动5V继电器模拟开锁,错误则触发有源蜂鸣器报警,有效保障了安全控制与实时警示。整体而言,该系统结合了多种硬件模块,实现了高效、安全的密码锁控制,适用于家庭或工业场景,展现了单片机在嵌入式应用中的实用性与扩展潜力。
  • [技术干货] 基于单片机的CAN总线车辆信息监测仪
    项目开发背景随着汽车工业的快速发展,车辆电子系统日益复杂,CAN总线作为汽车内部通信的核心协议,被广泛应用于传输各种传感器和控制单元的数据,以实现高效可靠的车辆控制。实时监测车辆关键参数如车速、发动机转速和冷却液温度,不仅有助于驾驶员及时了解车辆状态,还能有效预防潜在故障,提升行车安全性和车辆性能。然而,传统的车辆诊断工具往往依赖专业设备,成本较高且操作不便,无法满足普通车主在日常使用中的实时监控需求。在实际驾驶过程中,车辆可能出现水温过高等异常情况,若未能及时发现和处理,容易导致发动机损坏或其他安全隐患。因此,开发一种低成本、便携式的车辆信息监测仪,能够通过CAN总线实时获取并解析数据,并通过直观的显示和报警功能提醒用户,具有重要的应用价值。本项目旨在利用单片机技术,结合CAN总线通信,设计一个集成化监测系统,以解决传统方法的局限性。基于STM32F103C8T6单片机的设计,充分利用其内置CAN控制器的高效数据处理能力,配合TJA1050收发器从汽车OBD-II接口读取信息,并通过OLED屏幕实时显示解析后的数据。同时,系统集成SD卡存储模块,可记录行车历史用于后续分析,而声光报警模块则能在检测到关键参数异常时及时发出警示。这种方案不仅降低了成本,还提高了系统的便携性和实用性。该监测仪的开发顺应了智能汽车和物联网技术的发展趋势,可广泛应用于个人车辆维护、车队管理以及汽车维修领域,为提升车辆使用效率和安全性提供了一种创新解决方案。通过实时数据监控和存储分析,它有助于延长车辆寿命,减少故障发生率,并推动汽车电子系统的进一步优化。设计实现的功能(1) 通过CAN总线控制器从汽车OBD-II接口读取车速、发动机转速、冷却液温度等信息。(2) 将解析后的车辆数据实时显示在OLED屏幕上。(3) 当检测到关键参数异常(如水温过高)时,进行声光报警。(4) 可将一段时间的行车数据存入SD卡,用于后续分析。项目硬件模块组成(1)主控芯片:STM32F103C8T6单片机(自带CAN控制器)。(2)通信模块:TJA1050 CAN总线收发器。(3)显示模块:1.3英寸OLED显示屏。(4)存储模块:SD卡模块(SPI接口)。(5)报警模块:有源蜂鸣器与LED指示灯。设计意义该车辆信息监测仪的设计意义在于实现对车辆关键运行参数的实时监控与记录,通过CAN总线技术从OBD-II接口高效获取数据,如车速、发动机转速和冷却液温度等,帮助驾驶员即时了解车辆状态,从而提升行车安全性和操作便利性。异常报警功能能在水温过高等危险情况发生时及时发出声光警示,有效预防潜在故障,降低车辆损坏风险,保障驾驶安全。通过SD卡模块存储行车数据,该设计支持对历史信息的回溯与分析,为车辆维护和故障诊断提供可靠依据,有助于优化驾驶习惯和延长车辆使用寿命。硬件上采用STM32单片机和标准CAN收发器,确保了系统的稳定性和兼容性,可广泛应用于现代车辆监控场景,具有较高的实用价值和推广潜力。
  • [技术干货] 基于北斗的户外遇险预警定位装置
    项目开发背景随着户外探险、徒步旅行和野外作业等活动的日益普及,越来越多的人投身于自然环境中寻求挑战与乐趣。然而,这些活动往往伴随着较高的风险,如迷路、跌落、突发疾病或遭遇自然灾害等。在偏远地区,传统通信网络覆盖不足,导致遇险时难以及时求助,严重威胁用户生命安全。因此,开发一种高效可靠的户外遇险预警定位装置成为迫切需求,以提升户外活动的安全性和应急救援效率。北斗卫星导航系统作为中国自主建设的全球卫星导航系统,具有高精度、高可靠性和广域覆盖的特点,尤其适用于复杂地形和恶劣环境下的定位服务。结合北斗技术,可以确保用户在户外活动中实时获取准确的地理位置信息,为快速救援提供关键数据支撑。这一系统的发展为户外安全设备提供了坚实的技术基础,推动相关创新应用的涌现。本装置通过集成北斗定位、4G通信和姿态传感等功能,实现了对用户位置和身体状态的智能监控。用户可通过一键报警按钮主动发送精确的求救信息,而姿态解算功能则能在检测到剧烈跌落或长时间静止等异常情况时自动触发报警,弥补了人工操作的局限性。这种设计不仅提高了遇险响应的及时性,还增强了设备在紧急情况下的自主应对能力。为了适应户外长时间使用的场景,装置采用低功耗硬件设计,包括高性能主控芯片和大容量锂电池,确保在恶劣环境下仍能保持稳定运行和较长待机时间。这使得设备在徒步、登山、科考或应急救援等活动中具有广泛的适用性,为用户提供持续的安全保障。设计实现的功能(1) 集成北斗定位模块,实时获取并记录用户的地理位置坐标。(2) 配备一键报警按钮,遇险时按下可将包含精确位置的求救信息通过4G网络发送给预设联系人。(3) 具备姿态解算功能,当检测到使用者发生剧烈跌落或长时间静止不动时,触发自动报警。(4) 装置待机时间长,适合户外长时间使用。项目硬件模块组成(1) 主控芯片:STM32F103C8T6单片机。(2) 定位与通信:ATGM336H北斗/GPS双模定位模块、SIM7600CE 4G Cat1通信模块。(3) 姿态传感:MPU6050六轴传感器。(4) 输入与供电:物理报警按钮、大容量18650锂电池。设计意义该装置通过集成北斗定位模块和4G通信功能,能够在用户遇险时实时获取并发送精确地理位置,显著提高户外活动中的安全性和救援效率。这确保了在紧急情况下,用户能够迅速向预设联系人发出求救信号,缩短救援响应时间,为户外探险者提供关键的生命保障。姿态解算功能通过MPU6050六轴传感器检测使用者的运动状态,当发生剧烈跌落或长时间静止不动时自动触发报警,提供额外的安全保障,防止因意外事故而无法手动报警的情况,增强了设备的主动防护能力。装置采用大容量锂电池和低功耗设计,确保长时间待机,适合户外探险、徒步等长时间活动使用,增强了设备的实用性和可靠性,满足用户在偏远或无人区域对持久电力的需求。整体设计结合了先进的定位、通信和传感技术,为户外爱好者提供了一个可靠的安全伴侣,有助于减少户外活动中的风险,提升用户体验,同时推动户外安全设备的智能化发展。设计思路本装置的设计以STM32F103C8T6单片机作为核心控制器,负责协调和管理各个硬件模块的运行。该主控芯片具备高效的处理能力和低功耗特性,能够实时处理定位数据、姿态传感器信息以及通信任务,确保系统稳定可靠。通过集成ATGM336H北斗/GPS双模定位模块,装置能够持续获取用户的地理位置坐标,并将数据记录在系统中,为后续报警功能提供精确的位置信息。在通信方面,装置采用SIM7600CE 4G Cat1模块实现无线传输功能。当用户遇到险情时,可通过物理报警按钮手动触发求救,主控芯片会立即将包含当前位置的报警信息通过4G网络发送给预设的联系人。同时,该模块支持高效的数据传输,确保求救信息及时送达,增强了户外使用的安全性。姿态检测功能通过MPU6050六轴传感器实现,该传感器能够实时监测用户的加速度和角速度变化。主控芯片对传感器数据进行解算和分析,当检测到剧烈跌落或长时间静止不动等异常状态时,系统会自动触发报警机制,无需用户手动干预,从而提高了应对突发情况的响应能力。电源管理部分采用大容量18650锂电池供电,结合主控芯片的低功耗设计,优化了整体能耗。通过合理的电源调度和休眠机制,装置在待机状态下能够长时间运行,适合户外探险等场景的持续使用,确保用户在需要时随时可用。用户交互设计简单直观,物理报警按钮直接连接到主控芯片,便于在紧急情况下快速操作。整个系统注重实用性和可靠性,各模块协同工作,实现了定位、报警和姿态监测的核心功能,满足户外遇险预警的需求。框架图+-------------------+ +-------------------+ +-------------------+ | | | | | | | 北斗/GPS定位模块 |------>| 主控芯片 |<------| MPU6050姿态传感器 | | (ATGM336H) | | (STM32F103C8T6) | | | | | | | | | +-------------------+ +-------------------+ +-------------------+ | | v +-------------------+ +-------------------+ | | | | | 4G通信模块 |------>| 预设联系人 | | (SIM7600CE) | | (通过4G网络) | | | | | +-------------------+ +-------------------+ ^ | +-------------------+ | | | 物理报警按钮 | | | +-------------------+ | +-------------------+ | | | 18650锂电池供电 | | | +-------------------+ 系统总体设计系统总体设计围绕北斗户外遇险预警定位装置的功能需求和硬件模块展开,以STM32F103C8T6单片机作为核心控制器,协调各模块的运作。该系统旨在为户外活动提供可靠的遇险预警和定位服务,通过集成北斗定位、4G通信和姿态传感功能,实现实时监控和快速响应。系统通过ATGM336H北斗/GPS双模定位模块实时获取用户的地理位置坐标,并将数据记录在主控芯片中。同时,SIM7600CE 4G Cat1通信模块负责数据传输,确保在遇险时能够通过4G网络将包含精确位置的求救信息发送给预设联系人。这种设计保证了定位和通信的稳定性和实时性。报警机制包括手动和自动两种方式:用户可通过物理报警按钮一键触发报警,而MPU6050六轴传感器则用于姿态解算,当检测到剧烈跌落或长时间静止不动时,系统会自动判断为遇险状态并启动报警流程。这增强了装置的主动预警能力,适应户外复杂环境。电源部分采用大容量18650锂电池供电,结合低功耗设计,确保装置具有较长的待机时间,适合户外长时间使用。整个系统通过硬件模块的协同工作,实现了高效、可靠的遇险预警和定位功能。系统功能总结功能描述实时定位集成北斗定位模块,实时获取并记录用户的地理位置坐标。一键报警配备一键报警按钮,遇险时按下可将包含精确位置的求救信息通过4G网络发送给预设联系人。自动报警具备姿态解算功能,当检测到使用者发生剧烈跌落或长时间静止不动时,触发自动报警。长待机装置待机时间长,适合户外长时间使用。设计的各个功能模块描述主控芯片模块采用STM32F103C8T6单片机作为核心控制器,负责协调系统中各个模块的运行。它处理来自定位模块的地理位置数据、姿态传感器的运动信息,并控制通信模块在必要时发送报警信号。同时,该芯片管理电源分配,以优化能耗,确保装置在户外长时间稳定工作。定位与通信模块集成ATGM336H北斗/GPS双模定位模块,用于实时获取并记录用户的精确地理位置坐标,提高定位的可靠性和覆盖范围。SIM7600CE 4G Cat1通信模块则负责在用户触发报警时,通过4G网络将包含位置信息的求救消息快速发送给预设的联系人,实现高效的远程通信。姿态传感模块使用MPU6050六轴传感器监测使用者的运动状态,通过内置的姿态解算功能分析加速度和角速度数据。当检测到剧烈跌落或长时间静止不动等异常情况时,该模块会向主控芯片发送信号,自动触发报警流程,增强户外安全防护。输入与供电模块包括物理报警按钮和大容量18650锂电池。报警按钮提供一键报警功能,用户在遇险时可手动启动求救。锂电池为整个装置供电,通过高效能源管理确保长待机时间,适应户外活动的持久使用需求。上位机代码设计#include <iostream> #include <string> #include <vector> #include <thread> #include <chrono> #include <mutex> #include <map> #include <fstream> #include <sstream> #include <iomanip> #include <winsock2.h> #include <ws2tcpip.h> #pragma comment(lib, "ws2_32.lib") // 设备数据结构 struct DeviceData { std::string deviceId; double latitude; double longitude; std::string alertType; // "manual", "fall", "static" std::string timestamp; bool isOnline; std::chrono::system_clock::time_point lastUpdate; }; class OutdoorAlertSystem { private: std::map<std::string, DeviceData> devices; std::mutex dataMutex; SOCKET serverSocket; bool isRunning; int serverPort; public: OutdoorAlertSystem(int port = 8080) : serverPort(port), isRunning(false) {} ~OutdoorAlertSystem() { stopServer(); } // 初始化服务器 bool initializeServer() { WSADATA wsaData; if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) { std::cerr << "WSAStartup failed" << std::endl; return false; } serverSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (serverSocket == INVALID_SOCKET) { std::cerr << "Socket creation failed" << std::endl; WSACleanup(); return false; } sockaddr_in serverAddr; serverAddr.sin_family = AF_INET; serverAddr.sin_addr.s_addr = INADDR_ANY; serverAddr.sin_port = htons(serverPort); if (bind(serverSocket, (sockaddr*)&serverAddr, sizeof(serverAddr)) == SOCKET_ERROR) { std::cerr << "Bind failed" << std::endl; closesocket(serverSocket); WSACleanup(); return false; } if (listen(serverSocket, SOMAXCONN) == SOCKET_ERROR) { std::cerr << "Listen failed" << std::endl; closesocket(serverSocket); WSACleanup(); return false; } std::cout << "Server started on port " << serverPort << std::endl; return true; } // 解析设备数据 DeviceData parseDeviceData(const std::string& data) { DeviceData device; std::istringstream iss(data); std::string token; std::getline(iss, device.deviceId, ','); std::getline(iss, token, ','); device.latitude = std::stod(token); std::getline(iss, token, ','); device.longitude = std::stod(token); std::getline(iss, device.alertType, ','); std::getline(iss, device.timestamp); device.isOnline = true; device.lastUpdate = std::chrono::system_clock::now(); return device; } // 处理客户端连接 void handleClient(SOCKET clientSocket) { char buffer[1024]; int bytesReceived; while ((bytesReceived = recv(clientSocket, buffer, sizeof(buffer) - 1, 0)) > 0) { buffer[bytesReceived] = '\0'; std::string data(buffer); std::lock_guard<std::mutex> lock(dataMutex); DeviceData device = parseDeviceData(data); devices[device.deviceId] = device; // 报警处理 if (device.alertType != "normal") { handleAlert(device); } // 发送确认响应 std::string response = "ACK"; send(clientSocket, response.c_str(), response.length(), 0); } closesocket(clientSocket); } // 处理报警 void handleAlert(const DeviceData& device) { std::cout << "\n=== ALERT RECEIVED ===" << std::endl; std::cout << "Device ID: " << device.deviceId << std::endl; std::cout << "Alert Type: " << device.alertType << std::endl; std::cout << "Location: " << std::fixed << std::setprecision(6) << device.latitude << ", " << device.longitude << std::endl; std::cout << "Time: " << device.timestamp << std::endl; std::cout << "========================\n" << std::endl; // 记录报警日志 logAlert(device); // 这里可以添加发送邮件、短信等通知功能 // sendEmergencyNotification(device); } // 记录报警日志 void logAlert(const DeviceData& device) { std::ofstream logFile("alerts.log", std::ios::app); if (logFile.is_open()) { auto now = std::chrono::system_clock::now(); auto time_t = std::chrono::system_clock::to_time_t(now); logFile << "[" << std::put_time(std::localtime(&time_t), "%Y-%m-%d %H:%M:%S") << "] " << "Device: " << device.deviceId << " | " << "Type: " << device.alertType << " | " << "Location: " << device.latitude << ", " << device.longitude << std::endl; logFile.close(); } } // 设备状态监控线程 void deviceMonitorThread() { while (isRunning) { std::this_thread::sleep_for(std::chrono::seconds(30)); std::lock_guard<std::mutex> lock(dataMutex); auto now = std::chrono::system_clock::now(); for (auto& pair : devices) { auto& device = pair.second; auto duration = std::chrono::duration_cast<std::chrono::seconds>( now - device.lastUpdate); if (duration.count() > 60) { // 超过60秒无更新视为离线 device.isOnline = false; } } } } // 显示设备状态 void displayDeviceStatus() { std::lock_guard<std::mutex> lock(dataMutex); std::cout << "\n=== Device Status ===" << std::endl; std::cout << std::left << std::setw(15) << "Device ID" << std::setw(10) << "Status" << std::setw(20) << "Location" << std::setw(15) << "Last Alert" << std::setw(20) << "Last Update" << std::endl; std::cout << std::string(80, '-') << std::endl; for (const auto& pair : devices) { const auto& device = pair.second; std::string status = device.isOnline ? "Online" : "Offline"; std::string location = std::to_string(device.latitude).substr(0, 8) + ", " + std::to_string(device.longitude).substr(0, 8); auto time_t = std::chrono::system_clock::to_time_t(device.lastUpdate); std::string lastUpdate = std::put_time(std::localtime(&time_t), "%H:%M:%S"); std::cout << std::left << std::setw(15) << device.deviceId << std::setw(10) << status << std::setw(20) << location << std::setw(15) << device.alertType << std::setw(20) << lastUpdate << std::endl; } std::cout << std::endl; } // 启动服务器 void startServer() { if (!initializeServer()) { return; } isRunning = true; // 启动设备监控线程 std::thread monitorThread(&OutdoorAlertSystem::deviceMonitorThread, this); monitorThread.detach(); // 启动状态显示线程 std::thread displayThread([this]() { while (isRunning) { std::this_thread::sleep_for(std::chrono::seconds(10)); displayDeviceStatus(); } }); displayThread.detach(); // 主接收循环 while (isRunning) { sockaddr_in clientAddr; int clientAddrSize = sizeof(clientAddr); SOCKET clientSocket = accept(serverSocket, (sockaddr*)&clientAddr, &clientAddrSize); if (clientSocket == INVALID_SOCKET) { if (isRunning) { std::cerr << "Accept failed" << std::endl; } continue; } char clientIP[INET_ADDRSTRLEN]; inet_ntop(AF_INET, &clientAddr.sin_addr, clientIP, INET_ADDRSTRLEN); std::cout << "Client connected: " << clientIP << std::endl; // 为每个客户端创建新线程 std::thread clientThread(&OutdoorAlertSystem::handleClient, this, clientSocket); clientThread.detach(); } } // 停止服务器 void stopServer() { isRunning = false; if (serverSocket != INVALID_SOCKET) { closesocket(serverSocket); serverSocket = INVALID_SOCKET; } WSACleanup(); } // 导出设备数据到文件 void exportDeviceData(const std::string& filename) { std::lock_guard<std::mutex> lock(dataMutex); std::ofstream file(filename); if (file.is_open()) { file << "DeviceID,Latitude,Longitude,AlertType,LastUpdate,Status\n"; for (const auto& pair : devices) { const auto& device = pair.second; auto time_t = std::chrono::system_clock::to_time_t(device.lastUpdate); file << device.deviceId << "," << device.latitude << "," << device.longitude << "," << device.alertType << "," << std::put_time(std::localtime(&time_t), "%Y-%m-%d %H:%M:%S") << "," << (device.isOnline ? "Online" : "Offline") << "\n"; } file.close(); std::cout << "Device data exported to " << filename << std::endl; } } }; int main() { OutdoorAlertSystem server(8080); std::cout << "北斗户外遇险预警定位系统 - 上位机软件" << std::endl; std::cout << "=====================================" << std::endl; // 启动服务器 std::thread serverThread([&server]() { server.startServer(); }); // 主线程处理用户输入 std::string command; while (true) { std::cout << "\nCommands: [status] [export] [quit]" << std::endl; std::cout << "Enter command: "; std::getline(std::cin, command); if (command == "quit" || command == "exit") { break; } else if (command == "status") { server.displayDeviceStatus(); } else if (command == "export") { server.exportDeviceData("device_data.csv"); } } server.stopServer(); if (serverThread.joinable()) { serverThread.join(); } std::cout << "Server stopped." << std::endl; return 0; } 模块代码设计#include "stm32f10x.h" // MPU6050寄存器定义 #define MPU6050_ADDR 0xD0 #define SMPLRT_DIV 0x19 #define CONFIG 0x1A #define GYRO_CONFIG 0x1B #define ACCEL_CONFIG 0x1C #define ACCEL_XOUT_H 0x3B #define ACCEL_XOUT_L 0x3C #define ACCEL_YOUT_H 0x3D #define ACCEL_YOUT_L 0x3E #define ACCEL_ZOUT_H 0x3F #define ACCEL_ZOUT_L 0x40 #define GYRO_XOUT_H 0x43 #define GYRO_XOUT_L 0x44 #define GYRO_YOUT_H 0x45 #define GYRO_YOUT_L 0x46 #define GYRO_ZOUT_H 0x47 #define GYRO_ZOUT_L 0x48 #define PWR_MGMT_1 0x6B #define WHO_AM_I 0x75 // 硬件定义 #define BUTTON_PIN GPIO_Pin_0 #define LED_PIN GPIO_Pin_13 // 全局变量 volatile uint8_t alarm_triggered = 0; int16_t Accel_X, Accel_Y, Accel_Z; int16_t Gyro_X, Gyro_Y, Gyro_Z; // 延时函数 void Delay(uint32_t nCount) { for(; nCount != 0; nCount--); } // I2C初始化 void I2C1_Init(void) { RCC->APB1ENR |= RCC_APB1ENR_I2C1EN; RCC->APB2ENR |= RCC_APB2ENR_IOPBEN; GPIOB->CRL &= ~(GPIO_CRL_CNF6 | GPIO_CRL_CNF7); GPIOB->CRL |= GPIO_CRL_CNF6_1 | GPIO_CRL_CNF7_1; GPIOB->CRL |= GPIO_CRL_MODE6 | GPIO_CRL_MODE7; I2C1->CR1 &= ~I2C_CR1_PE; I2C1->CR2 = 36; I2C1->CCR = 180; I2C1->TRISE = 37; I2C1->CR1 |= I2C_CR1_PE; } // I2C起始信号 void I2C1_Start(void) { I2C1->CR1 |= I2C_CR1_START; while(!(I2C1->SR1 & I2C_SR1_SB)); } // I2C停止信号 void I2C1_Stop(void) { I2C1->CR1 |= I2C_CR1_STOP; while(I2C1->CR1 & I2C_CR1_STOP); } // I2C发送地址 void I2C1_Address(uint8_t address) { I2C1->DR = address; while(!(I2C1->SR1 & I2C_SR1_ADDR)); (void)I2C1->SR2; } // I2C发送数据 void I2C1_Write(uint8_t data) { I2C1->DR = data; while(!(I2C1->SR1 & I2C_SR1_TXE)); } // I2C读取数据 uint8_t I2C1_Read(void) { while(!(I2C1->SR1 & I2C_SR1_RXNE)); return I2C1->DR; } // MPU6050写寄存器 void MPU6050_WriteReg(uint8_t reg, uint8_t data) { I2C1_Start(); I2C1_Address(MPU6050_ADDR); I2C1_Write(reg); I2C1_Write(data); I2C1_Stop(); } // MPU6050读寄存器 uint8_t MPU6050_ReadReg(uint8_t reg) { uint8_t data; I2C1_Start(); I2C1_Address(MPU6050_ADDR); I2C1_Write(reg); I2C1_Start(); I2C1_Address(MPU6050_ADDR | 0x01); data = I2C1_Read(); I2C1_Stop(); return data; } // MPU6050初始化 void MPU6050_Init(void) { MPU6050_WriteReg(PWR_MGMT_1, 0x80); Delay(100000); MPU6050_WriteReg(PWR_MGMT_1, 0x00); MPU6050_WriteReg(SMPLRT_DIV, 0x07); MPU6050_WriteReg(CONFIG, 0x06); MPU6050_WriteReg(GYRO_CONFIG, 0x18); MPU6050_WriteReg(ACCEL_CONFIG, 0x01); } // 读取MPU6050数据 void MPU6050_ReadData(void) { I2C1_Start(); I2C1_Address(MPU6050_ADDR); I2C1_Write(ACCEL_XOUT_H); I2C1_Start(); I2C1_Address(MPU6050_ADDR | 0x01); Accel_X = (I2C1_Read() << 8) | I2C1_Read(); Accel_Y = (I2C1_Read() << 8) | I2C1_Read(); Accel_Z = (I2C1_Read() << 8) | I2C1_Read(); Gyro_X = (I2C1_Read() << 8) | I2C1_Read(); Gyro_Y = (I2C1_Read() << 8) | I2C1_Read(); Gyro_Z = (I2C1_Read() << 8) | I2C1_Read(); I2C1_Stop(); } // USART1初始化 (GPS模块) void USART1_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_USART1EN | RCC_APB2ENR_IOPAEN; GPIOA->CRH &= ~(GPIO_CRH_CNF9 | GPIO_CRH_CNF10); GPIOA->CRH |= GPIO_CRH_CNF9_1 | GPIO_CRH_CNF10_0; GPIOA->CRH |= GPIO_CRH_MODE9 | GPIO_CRH_MODE10; USART1->BRR = 0x1D4C; USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; } // USART2初始化 (4G模块) void USART2_Init(void) { RCC->APB1ENR |= RCC_APB1ENR_USART2EN; RCC->APB2ENR |= RCC_APB2ENR_IOPAEN; GPIOA->CRL &= ~(GPIO_CRL_CNF2 | GPIO_CRL_CNF3); GPIOA->CRL |= GPIO_CRL_CNF2_1 | GPIO_CRL_CNF3_0; GPIOA->CRL |= GPIO_CRL_MODE2 | GPIO_CRL_MODE3; USART2->BRR = 0x1D4C; USART2->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; } // USART发送字符串 void USART_SendString(USART_TypeDef* USARTx, char* str) { while(*str) { while(!(USARTx->SR & USART_SR_TXE)); USARTx->DR = *str++; } } // GPIO初始化 void GPIO_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPCEN; // 报警按钮输入 GPIOA->CRL &= ~GPIO_CRL_CNF0; GPIOA->CRL |= GPIO_CRL_CNF0_1; // LED输出 GPIOC->CRH &= ~GPIO_CRH_CNF13; GPIOC->CRH |= GPIO_CRH_MODE13; } // 外部中断初始化 void EXTI_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_AFIOEN; AFIO->EXTICR[0] |= AFIO_EXTICR1_EXTI0_PA; EXTI->IMR |= EXTI_IMR_MR0; EXTI->FTSR |= EXTI_FTSR_TR0; NVIC->ISER[0] |= 1 << 6; } // 姿态检测函数 uint8_t DetectFall(void) { static uint32_t stationary_time = 0; float accel_magnitude = sqrt(Accel_X*Accel_X + Accel_Y*Accel_Y + Accel_Z*Accel_Z); // 检测剧烈跌落 if(accel_magnitude > 16000 || accel_magnitude < 8000) { return 1; } // 检测长时间静止 if(abs(Gyro_X) < 100 && abs(Gyro_Y) < 100 && abs(Gyro_Z) < 100) { stationary_time++; if(stationary_time > 300000) { // 约5分钟 stationary_time = 0; return 1; } } else { stationary_time = 0; } return 0; } // 发送报警信息 void SendAlarm(void) { char alarm_msg[] = "SOS! Emergency! Location: "; USART_SendString(USART2, "AT+CMGF=1\r\n"); Delay(100000); USART_SendString(USART2, "AT+CMGS=\"+86138xxxxyyyy\"\r\n"); Delay(100000); USART_SendString(USART2, alarm_msg); // 这里添加GPS坐标信息 USART_SendString(USART2, "\x1A"); } // 外部中断服务函数 void EXTI0_IRQHandler(void) { if(EXTI->PR & EXTI_PR_PR0) { alarm_triggered = 1; EXTI->PR |= EXTI_PR_PR0; } } int main(void) { SystemInit(); GPIO_Init(); I2C1_Init(); USART1_Init(); USART2_Init(); EXTI_Init(); MPU6050_Init(); while(1) { MPU6050_ReadData(); // 检测姿态异常 if(DetectFall()) { alarm_triggered = 1; } // 触发报警 if(alarm_triggered) { GPIOC->BSRR = LED_PIN; SendAlarm(); alarm_triggered = 0; Delay(1000000); GPIOC->BRR = LED_PIN; } Delay(10000); } } 项目核心代码#include "stm32f10x.h" // 硬件模块定义 #define BUTTON_GPIO_PORT GPIOA #define BUTTON_PIN GPIO_Pin_0 #define LED_GPIO_PORT GPIOC #define LED_PIN GPIO_Pin_13 // 模块初始化状态 typedef enum { MODULE_INIT_SUCCESS = 0, MODULE_INIT_FAIL = 1 } Module_Status; // 报警状态 typedef enum { ALARM_OFF = 0, ALARM_BUTTON = 1, ALARM_FALL = 2, ALARM_STATIC = 3 } Alarm_State; // 全局变量 volatile uint8_t alarm_triggered = 0; volatile Alarm_State alarm_type = ALARM_OFF; volatile uint32_t system_tick = 0; // 假设的外部模块函数声明 extern Module_Status GPS_Init(void); extern Module_Status GPRS_Init(void); extern Module_Status MPU6050_Init(void); extern Module_Status Battery_Init(void); extern void GPS_GetPosition(float *latitude, float *longitude); extern uint8_t GPRS_SendAlert(float lat, float lon, Alarm_State type); extern uint8_t MPU6050_DetectFall(void); extern uint8_t MPU6050_DetectStatic(void); extern float Battery_GetVoltage(void); // 系统时钟初始化 void RCC_Configuration(void) { // 开启HSE RCC->CR |= ((uint32_t)RCC_CR_HSEON); // 等待HSE就绪 while(!(RCC->CR & RCC_CR_HSERDY)); // FLASH预取指缓冲使能 FLASH->ACR |= FLASH_ACR_PRFTBE; // 设置等待周期 FLASH->ACR &= (uint32_t)((uint32_t)~FLASH_ACR_LATENCY); FLASH->ACR |= (uint32_t)FLASH_ACR_LATENCY_2; // HCLK = SYSCLK RCC->CFGR |= (uint32_t)RCC_CFGR_HPRE_DIV1; // PCLK2 = HCLK RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE2_DIV1; // PCLK1 = HCLK/2 RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE1_DIV2; // PLL配置: PLLCLK = HSE * 9 = 72 MHz RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_PLLSRC | RCC_CFGR_PLLXTPRE | RCC_CFGR_PLLMULL)); RCC->CFGR |= (uint32_t)(RCC_CFGR_PLLSRC_HSE | RCC_CFGR_PLLMULL9); // 开启PLL RCC->CR |= RCC_CR_PLLON; // 等待PLL就绪 while(!(RCC->CR & RCC_CR_PLLRDY)); // 选择PLL作为系统时钟 RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_SW)); RCC->CFGR |= (uint32_t)RCC_CFGR_SW_PLL; // 等待时钟切换完成 while ((RCC->CFGR & (uint32_t)RCC_CFGR_SWS) != (uint32_t)0x08); } // GPIO初始化 void GPIO_Configuration(void) { // 开启GPIO时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPCEN; // 报警按钮配置(PA0输入) BUTTON_GPIO_PORT->CRL &= ~(GPIO_CRL_MODE0 | GPIO_CRL_CNF0); BUTTON_GPIO_PORT->CRL |= GPIO_CRL_CNF0_1; // 浮空输入 // LED指示灯配置(PC13推挽输出) LED_GPIO_PORT->CRH &= ~(GPIO_CRH_MODE13 | GPIO_CRH_CNF13); LED_GPIO_PORT->CRH |= GPIO_CRH_MODE13_1; // 输出模式,最大速度2MHz } // SysTick定时器初始化 void SysTick_Init(void) { // 重装载值,1ms中断一次 SysTick->LOAD = 72000 - 1; // 设置当前值 SysTick->VAL = 0; // 设置时钟源,使能中断,启动定时器 SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk | SysTick_CTRL_TICKINT_Msk | SysTick_CTRL_ENABLE_Msk; } // 外部中断初始化(报警按钮) void EXTI_Configuration(void) { // 开启AFIO时钟 RCC->APB2ENR |= RCC_APB2ENR_AFIOEN; // PA0连接到EXTI0 AFIO->EXTICR[0] &= ~AFIO_EXTICR1_EXTI0; AFIO->EXTICR[0] |= AFIO_EXTICR1_EXTI0_PA; // 配置EXTI0为下降沿触发 EXTI->FTSR |= EXTI_FTSR_TR0; // 使能EXTI0中断 EXTI->IMR |= EXTI_IMR_MR0; // 配置NVIC NVIC_EnableIRQ(EXTI0_IRQn); NVIC_SetPriority(EXTI0_IRQn, 0x00); } // LED控制 void LED_Control(uint8_t state) { if(state) { LED_GPIO_PORT->BSRR = LED_PIN; // 置位 } else { LED_GPIO_PORT->BRR = LED_PIN; // 复位 } } // 系统延时函数 void Delay_ms(uint32_t ms) { uint32_t start_tick = system_tick; while((system_tick - start_tick) < ms); } // 报警处理函数 void Handle_Alarm(Alarm_State alarm_type) { float latitude, longitude; // 获取当前位置 GPS_GetPosition(&latitude, &longitude); // 发送报警信息 if(GPRS_SendAlert(latitude, longitude, alarm_type)) { // 发送成功,LED闪烁3次 for(int i = 0; i < 3; i++) { LED_Control(1); Delay_ms(200); LED_Control(0); Delay_ms(200); } } else { // 发送失败,持续闪烁 while(1) { LED_Control(1); Delay_ms(100); LED_Control(0); Delay_ms(100); } } } // 系统初始化 uint8_t System_Init(void) { // 初始化系统时钟 RCC_Configuration(); // 初始化GPIO GPIO_Configuration(); // 初始化SysTick SysTick_Init(); // 初始化外部中断 EXTI_Configuration(); // 初始化各功能模块 if(GPS_Init() != MODULE_INIT_SUCCESS) return 0; if(GPRS_Init() != MODULE_INIT_SUCCESS) return 0; if(MPU6050_Init() != MODULE_INIT_SUCCESS) return 0; if(Battery_Init() != MODULE_INIT_SUCCESS) return 0; // 开机指示灯 LED_Control(1); Delay_ms(1000); LED_Control(0); return 1; } // 主函数 int main(void) { // 系统初始化 if(!System_Init()) { // 初始化失败,LED常亮 LED_Control(1); while(1); } uint32_t last_position_update = 0; uint32_t last_sensor_check = 0; uint32_t last_battery_check = 0; while(1) { // 检查报警触发 if(alarm_triggered) { Handle_Alarm(alarm_type); alarm_triggered = 0; alarm_type = ALARM_OFF; } // 每30秒更新一次位置信息 if((system_tick - last_position_update) > 30000) { float lat, lon; GPS_GetPosition(&lat, &lon); last_position_update = system_tick; // LED闪烁指示正常工作 LED_Control(1); Delay_ms(50); LED_Control(0); } // 每2秒检查一次姿态传感器 if((system_tick - last_sensor_check) > 2000) { // 检测跌落 if(MPU6050_DetectFall()) { alarm_triggered = 1; alarm_type = ALARM_FALL; } // 检测长时间静止 else if(MPU6050_DetectStatic()) { alarm_triggered = 1; alarm_type = ALARM_STATIC; } last_sensor_check = system_tick; } // 每5分钟检查一次电池电量 if((system_tick - last_battery_check) > 300000) { float voltage = Battery_GetVoltage(); // 低电量处理(可根据需要扩展) if(voltage < 3.3f) { // 低电量警告 for(int i = 0; i < 5; i++) { LED_Control(1); Delay_ms(100); LED_Control(0); Delay_ms(100); } } last_battery_check = system_tick; } // 进入低功耗模式(根据实际需求实现) // __WFI(); } } // SysTick中断服务函数 void SysTick_Handler(void) { system_tick++; } // 外部中断服务函数(报警按钮) void EXTI0_IRQHandler(void) { if(EXTI->PR & EXTI_PR_PR0) { // 清除中断标志 EXTI->PR = EXTI_PR_PR0; // 触发按钮报警 alarm_triggered = 1; alarm_type = ALARM_BUTTON; } } 总结本装置是一款专为户外活动设计的遇险预警定位设备,依托北斗系统实现精准定位和紧急通信功能,旨在保障用户在偏远或危险环境中的安全。在功能上,它能够实时获取并记录用户的地理位置坐标,配备一键报警按钮,可在遇险时快速发送包含精确位置的求救信息至预设联系人,同时通过姿态解算自动检测剧烈跌落或长时间静止状态,及时触发报警机制。硬件方面,装置采用STM32F103C8T6单片机作为主控核心,集成ATGM336H北斗/GPS双模定位模块和SIM7600CE 4G通信模块,确保定位与数据传输的可靠性,并搭载MPU6050六轴传感器进行姿态监测,辅以物理报警按钮和大容量18650锂电池供电。整体设计注重实用性和耐用性,待机时间长,适合户外长时间使用,为用户提供全天候的遇险预警和定位保障。
  • [技术干货] 基于Zigbee的智能家居灯光控制系统
    项目开发背景随着科技的飞速发展和人们生活水平的提高,智能家居系统逐渐成为现代家庭的重要组成部分。智能灯光控制作为智能家居的核心功能之一,不仅能够提升居住舒适度,还能有效节约能源,满足用户对便捷、智能化生活的追求。传统的灯光控制系统多依赖于有线连接或简单的遥控器,存在布线复杂、灵活性差、无法远程控制等局限性,用户往往需要手动操作开关,无法根据环境变化自动调节,也无法实现多房间的统一管理。为了克服这些不足,无线通信技术如Zigbee被引入智能家居领域。Zigbee技术以其低功耗、低成本、高可靠性和自组网能力,成为构建智能灯光控制系统的理想选择。它能够轻松组建包含协调器、路由器和终端设备的网络,实现设备间的稳定通信和广泛覆盖,从而支持灵活的远程控制和自动化管理。本项目的开发旨在利用Zigbee技术构建一个高效的智能家居灯光控制系统,通过集成STM32单片机、Zigbee模块、光敏电阻和继电器等硬件,实现环境光照监测、灯具开关控制、手机APP远程操作以及多种场景模式功能。这不仅解决了传统系统在灵活性和智能化方面的缺陷,还为推动智能家居的普及和能源节约提供了实用解决方案,具有重要的社会和经济价值。设计实现的功能(1) 组建Zigbee无线网络,包含协调器、路由器和终端设备。(2) 终端设备节点能监测环境光照强度,并控制灯具开关。(3) 用户可通过手机APP(通过协调器)远程控制任意房间的灯光。(4) 系统支持场景模式,如“全开”、“全关”、“夜间模式”。项目硬件模块组成(1) 主控与组网:STM32单片机 + Zigbee模块(如CC2530)。(2) 传感模块:光敏电阻模块。(3) 执行模块:继电器模块。(4) 协调器:Zigbee-USB协调器,用于连接上位机或网关。设计意义该设计基于Zigbee无线网络技术构建智能家居灯光控制系统,能够实现稳定可靠的设备组网,支持协调器、路由器和终端设备的协同工作,有效降低系统部署成本和维护复杂性,同时利用Zigbee的低功耗特性延长设备使用寿命,为家庭环境提供高效的无线通信基础。通过集成光敏电阻模块,系统能够实时监测环境光照强度,并结合继电器模块自动控制灯具开关,实现智能调光与节能管理,这不仅提升了能源利用效率,还增强了居住舒适度,满足用户对自动化照明的基本需求。系统支持用户通过手机APP远程控制任意房间的灯光,借助Zigbee-USB协调器连接上位机或网关,实现了灵活便捷的远程操作,大大提高了用户的生活便利性,尤其适用于外出或移动场景下的家庭照明管理。此外,系统内置多种场景模式如“全开”、“全关”和“夜间模式”,允许用户一键切换照明状态,简化了日常操作流程,增强了系统的实用性和智能化水平,为现代家庭提供了一种高效且易用的灯光控制解决方案。设计思路设计思路首先围绕Zigbee无线网络的组建展开,系统采用协调器作为网络核心,负责初始化和管理整个网络,路由器用于扩展网络覆盖范围并转发数据,终端设备则部署在各个房间中,通过Zigbee模块与协调器通信,确保网络稳定可靠。硬件上,协调器使用Zigbee-USB协调器连接上位机或网关,而终端设备以STM32单片机为主控,搭配Zigbee模块实现无线通信。在终端设备层面,每个节点集成光敏电阻模块来实时监测环境光照强度,并将数据通过Zigbee网络传输至协调器;同时,STM32单片机根据光照数据或控制指令驱动继电器模块,实现灯具的开关控制,确保系统能自动响应环境变化。这种设计使得终端设备既能独立工作,又能融入网络协同。用户远程控制功能通过手机APP实现,APP与协调器连接,协调器作为网关接收APP指令并将其转发至指定的终端设备节点,从而允许用户远程控制任意房间的灯光开关。系统通过协调器解析指令并路由到对应设备,确保控制实时性和准确性。场景模式功能基于协调器的逻辑处理实现,系统预设“全开”、“全关”和“夜间模式”等场景,协调器在接收到模式指令后,统一向所有相关终端设备发送批量控制命令,例如在“夜间模式”下自动调整灯光状态,提升用户体验和能源效率。整个系统设计注重实际应用,确保功能稳定且易于维护。框架图+-------------------+ +-------------------+ | 手机APP | | 协调器 | | (用户控制) |<----->| (Zigbee-USB) | | - 场景模式 | | - 网络管理 | | (全开/全关/夜间)| +-------------------+ +-------------------+ | | Zigbee 无线协议 | +-------------------------+-------------------------+ | | | +-------------+ +-------------+ +-------------+ | 路由器 | | 终端设备1 | | 终端设备2 | | (可选中继) | | (房间1) | | (房间2) | +-------------+ +-------------+ +-------------+ | | | | +-----------+ +-----------+ | |STM32+Zigbee| |STM32+Zigbee| | |光敏传感器 | |光敏传感器 | | |继电器 | |继电器 | | +-----------+ +-----------+ | | | | +-----------+ +-----------+ | | 灯具 | | 灯具 | | +-----------+ +-----------+ | | | +-------------+ +-------------+ +-------------+ | ...更多设备| | ...更多设备| | ...更多设备| +-------------+ +-------------+ +-------------+ 系统总体设计该系统总体设计基于Zigbee无线通信技术,构建一个智能家居灯光控制系统,实现灯光的远程控制和自动化管理。系统采用分布式架构,通过Zigbee网络连接各个设备节点,确保稳定可靠的数据传输和指令执行。Zigbee网络由协调器、路由器和终端设备组成,其中协调器作为网络中心,负责初始化网络并管理设备入网,路由器扩展网络覆盖范围,增强信号中继能力,终端设备则作为网络中的叶节点,执行具体的传感和控制任务。这种分层结构提高了系统的可扩展性和鲁棒性,适用于多房间家居环境。终端设备节点集成了光敏电阻模块,用于实时监测环境光照强度,并将数据通过Zigbee网络上报。同时,这些节点通过继电器模块控制灯具的开关状态,实现基于环境光的自动调节或用户指令的响应。每个终端设备可独立工作,也可通过网络协同操作。用户交互通过手机APP实现,APP通过协调器与Zigbee网络通信,协调器通常以Zigbee-USB形式连接至上位机或网关,作为无线网络与外部设备的接口。用户可通过APP远程控制任意房间的灯光,查看当前状态,并发送开关指令,确保便捷的远程管理。系统支持预设场景模式,如“全开”模式可一次性打开所有灯光,“全关”模式关闭所有灯光,“夜间模式”则根据光照阈值自动调整灯光状态,以优化能源使用和居住舒适度。这些模式通过协调器统一调度,终端设备协同执行,实现智能化的场景切换。硬件模块以STM32单片机作为主控制器,处理数据逻辑和通信协议,结合Zigbee模块如CC2530实现无线组网。光敏电阻模块负责环境光传感,继电器模块驱动灯具开关,Zigbee-USB协调器确保网络与外部系统的连接,整个硬件设计注重低功耗和实时性,满足家居应用需求。系统功能总结功能类别描述网络组建组建Zigbee无线网络,包含协调器、路由器和终端设备。环境监测与控制终端设备节点监测环境光照强度,并控制灯具开关。远程控制用户可通过手机APP(通过协调器)远程控制任意房间的灯光。场景模式系统支持场景模式,如“全开”、“全关”、“夜间模式”。设计的各个功能模块描述Zigbee通信模块负责组建无线网络,采用STM32单片机作为主控芯片,搭配Zigbee模块如CC2530实现网络功能。该模块包括协调器、路由器和终端设备,协调器负责网络初始化和数据转发,路由器扩展网络覆盖范围,终端设备连接传感器和执行器,确保网络稳定通信。环境监测模块集成光敏电阻模块于终端设备节点,用于实时检测环境光照强度。该模块将采集的光照数据通过Zigbee网络传输至协调器或相关控制单元,为自动灯光控制提供依据。灯光控制模块通过继电器模块实现对灯具的开关操作,继电器由STM32单片机驱动,根据接收到的控制指令或传感器数据执行相应动作,确保灯光状态可精确调整。远程控制模块允许用户通过手机APP发送指令,指令经由互联网传输至Zigbee-USB协调器,协调器将命令解析后通过Zigbee网络转发至目标终端设备,实现任意房间灯光的远程监控和管理。场景模式模块支持预设场景如“全开”、“全关”和“夜间模式”,用户可通过手机APP触发场景命令,系统自动协调所有终端设备执行相应操作,例如在夜间模式下根据光照强度自动调节灯光状态。上位机代码设计#include <QApplication> #include <QMainWindow> #include <QVBoxLayout> #include <QHBoxLayout> #include <QPushButton> #include <QLabel> #include <QComboBox> #include <QSerialPort> #include <QSerialPortInfo> #include <QMessageBox> #include <QTimer> #include <QGroupBox> #include <QTextEdit> #include <QSpinBox> #include <QCheckBox> #include <QProgressBar> class ZigbeeController : public QMainWindow { Q_OBJECT public: ZigbeeController(QWidget *parent = nullptr) : QMainWindow(parent) { setupUI(); setupSerialPort(); setupConnections(); } ~ZigbeeController() { if (serialPort && serialPort->isOpen()) { serialPort->close(); } } private slots: void connectToCoordinator() { if (serialPort) { serialPort->setPortName(portComboBox->currentText()); serialPort->setBaudRate(QSerialPort::Baud9600); serialPort->setDataBits(QSerialPort::Data8); serialPort->setParity(QSerialPort::NoParity); serialPort->setStopBits(QSerialPort::OneStop); serialPort->setFlowControl(QSerialPort::NoFlowControl); if (serialPort->open(QIODevice::ReadWrite)) { statusLabel->setText("已连接到协调器"); connectionStatus->setValue(100); refreshPorts(); } else { QMessageBox::critical(this, "连接错误", "无法连接到协调器"); } } } void refreshPorts() { portComboBox->clear(); QList<QSerialPortInfo> ports = QSerialPortInfo::availablePorts(); for (const QSerialPortInfo &port : ports) { portComboBox->addItem(port.portName()); } } void readData() { if (serialPort && serialPort->isOpen()) { QByteArray data = serialPort->readAll(); if (!data.isEmpty()) { QString message = QString::fromUtf8(data); logTextEdit->append("接收: " + message); processReceivedData(message); } } } void sendControlCommand(const QString &command) { if (serialPort && serialPort->isOpen()) { QByteArray data = command.toUtf8() + "\n"; serialPort->write(data); logTextEdit->append("发送: " + command); } else { QMessageBox::warning(this, "警告", "未连接到协调器"); } } void controlRoomLight() { QString room = roomComboBox->currentText(); QString command = room + "_" + (lightOnCheckBox->isChecked() ? "ON" : "OFF"); sendControlCommand(command); } void setBrightness() { QString room = roomComboBox->currentText(); int brightness = brightnessSpinBox->value(); QString command = room + "_BRIGHTNESS_" + QString::number(brightness); sendControlCommand(command); } void activateScene() { QString scene = sceneComboBox->currentText(); QString command = "SCENE_" + scene; sendControlCommand(command); } void queryNodeStatus() { sendControlCommand("QUERY_ALL_STATUS"); } private: void setupUI() { setWindowTitle("基于Zigbee的智能家居灯光控制系统"); setMinimumSize(800, 600); QWidget *centralWidget = new QWidget(this); setCentralWidget(centralWidget); QVBoxLayout *mainLayout = new QVBoxLayout(centralWidget); // 连接设置 QGroupBox *connectionGroup = new QGroupBox("协调器连接设置"); QHBoxLayout *connectionLayout = new QHBoxLayout(connectionGroup); portComboBox = new QComboBox(); refreshButton = new QPushButton("刷新端口"); connectButton = new QPushButton("连接协调器"); statusLabel = new QLabel("未连接"); connectionStatus = new QProgressBar(); connectionStatus->setRange(0, 100); connectionStatus->setValue(0); connectionLayout->addWidget(new QLabel("串口:")); connectionLayout->addWidget(portComboBox); connectionLayout->addWidget(refreshButton); connectionLayout->addWidget(connectButton); connectionLayout->addWidget(statusLabel); connectionLayout->addWidget(connectionStatus); // 房间灯光控制 QGroupBox *lightControlGroup = new QGroupBox("房间灯光控制"); QGridLayout *lightLayout = new QGridLayout(lightControlGroup); roomComboBox = new QComboBox(); roomComboBox->addItems({"客厅", "卧室", "厨房", "书房", "卫生间"}); lightOnCheckBox = new QCheckBox("开灯"); brightnessSpinBox = new QSpinBox(); brightnessSpinBox->setRange(0, 100); brightnessSpinBox->setValue(50); QPushButton *applyLightBtn = new QPushButton("应用设置"); QPushButton *queryStatusBtn = new QPushButton("查询状态"); lightLayout->addWidget(new QLabel("房间:"), 0, 0); lightLayout->addWidget(roomComboBox, 0, 1); lightLayout->addWidget(lightOnCheckBox, 1, 0); lightLayout->addWidget(new QLabel("亮度:"), 1, 1); lightLayout->addWidget(brightnessSpinBox, 1, 2); lightLayout->addWidget(applyLightBtn, 2, 0); lightLayout->addWidget(queryStatusBtn, 2, 1); // 场景模式 QGroupBox *sceneGroup = new QGroupBox("场景模式"); QHBoxLayout *sceneLayout = new QHBoxLayout(sceneGroup); sceneComboBox = new QComboBox(); sceneComboBox->addItems({"全开", "全关", "夜间模式", "会客模式", "影院模式"}); QPushButton *activateSceneBtn = new QPushButton("激活场景"); sceneLayout->addWidget(new QLabel("场景:")); sceneLayout->addWidget(sceneComboBox); sceneLayout->addWidget(activateSceneBtn); // 系统日志 QGroupBox *logGroup = new QGroupBox("系统日志"); QVBoxLayout *logLayout = new QVBoxLayout(logGroup); logTextEdit = new QTextEdit(); logTextEdit->setReadOnly(true); QPushButton *clearLogBtn = new QPushButton("清空日志"); logLayout->addWidget(logTextEdit); logLayout->addWidget(clearLogBtn); // 添加到主布局 mainLayout->addWidget(connectionGroup); mainLayout->addWidget(lightControlGroup); mainLayout->addWidget(sceneGroup); mainLayout->addWidget(logGroup); // 连接信号槽 connect(refreshButton, &QPushButton::clicked, this, &ZigbeeController::refreshPorts); connect(connectButton, &QPushButton::clicked, this, &ZigbeeController::connectToCoordinator); connect(applyLightBtn, &QPushButton::clicked, this, &ZigbeeController::controlRoomLight); connect(applyLightBtn, &QPushButton::clicked, this, &ZigbeeController::setBrightness); connect(queryStatusBtn, &QPushButton::clicked, this, &ZigbeeController::queryNodeStatus); connect(activateSceneBtn, &QPushButton::clicked, this, &ZigbeeController::activateScene); connect(clearLogBtn, &QPushButton::clicked, logTextEdit, &QTextEdit::clear); refreshPorts(); } void setupSerialPort() { serialPort = new QSerialPort(this); connect(serialPort, &QSerialPort::readyRead, this, &ZigbeeController::readData); } void setupConnections() { // 定时查询节点状态 QTimer *statusTimer = new QTimer(this); connect(statusTimer, &QTimer::timeout, this, &ZigbeeController::queryNodeStatus); statusTimer->start(10000); // 每10秒查询一次 } void processReceivedData(const QString &data) { // 解析接收到的数据 if (data.contains("LIGHT_INTENSITY")) { // 处理光照强度数据 QStringList parts = data.split(":"); if (parts.size() >= 2) { QString room = parts[0]; QString intensity = parts[1]; logTextEdit->append(room + " 光照强度: " + intensity); } } else if (data.contains("STATUS")) { // 处理设备状态 logTextEdit->append("设备状态更新: " + data); } } // UI组件 QComboBox *portComboBox; QComboBox *roomComboBox; QComboBox *sceneComboBox; QPushButton *refreshButton; QPushButton *connectButton; QCheckBox *lightOnCheckBox; QSpinBox *brightnessSpinBox; QLabel *statusLabel; QProgressBar *connectionStatus; QTextEdit *logTextEdit; // 串口通信 QSerialPort *serialPort; }; int main(int argc, char *argv[]) { QApplication app(argc, argv); ZigbeeController controller; controller.show(); return app.exec(); } #include "main.moc" # CMakeLists.txt cmake_minimum_required(VERSION 3.16) project(ZigbeeLightControl) set(CMAKE_CXX_STANDARD 14) set(CMAKE_CXX_STANDARD_REQUIRED ON) find_package(Qt6 REQUIRED COMPONENTS Core Widgets SerialPort) qt_standard_project_setup() qt_add_executable(ZigbeeLightControl main.cpp ) qt_add_resources(ZigbeeLightControl "resources" PREFIX "/" FILES icons/connect.png icons/disconnect.png icons/light_on.png icons/light_off.png ) target_link_libraries(ZigbeeLightControl Qt6::Core Qt6::Widgets Qt6::SerialPort) <!-- ZigbeeLightControl.pro --> QT += core widgets serialport CONFIG += c++17 SOURCES += \ main.cpp HEADERS += RESOURCES += \ resources.qrc这个完整的Zigbee智能家居灯光控制系统上位机代码包含以下主要功能:协调器连接管理:自动检测可用串口连接/断开Zigbee协调器连接状态显示房间灯光控制:选择不同房间控制灯光开关调节亮度等级实时状态查询场景模式控制:全开模式全关模式夜间模式会客模式影院模式系统监控:实时通信日志节点状态监控光照强度显示通信协议:基于串口的命令通信支持节点状态查询场景模式控制命令代码采用模块化设计,易于扩展和维护,支持跨平台运行。模块代码设计#include "stm32f10x.h" // 引脚定义 #define LIGHT_SENSOR_PIN GPIO_Pin_0 // PA0 - 光敏电阻 #define RELAY_PIN GPIO_Pin_13 // PC13 - 继电器控制 #define ZIGBEE_TX_PIN GPIO_Pin_9 // PA9 - Zigbee TX #define ZIGBEE_RX_PIN GPIO_Pin_10 // PA10 - Zigbee RX // ADC通道 #define LIGHT_SENSOR_ADC_CHANNEL ADC_Channel_0 // 系统变量 volatile uint16_t light_level = 0; volatile uint8_t light_status = 0; volatile uint8_t auto_mode = 1; // 函数声明 void RCC_Configuration(void); void GPIO_Configuration(void); void ADC_Configuration(void); void USART_Configuration(void); void NVIC_Configuration(void); uint16_t Read_ADC(uint8_t channel); void Delay_ms(uint32_t nTime); void Send_To_Zigbee(uint8_t* data, uint8_t len); void Process_Zigbee_Command(uint8_t* cmd); // 系统时钟配置 void RCC_Configuration(void) { // 开启外设时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPCEN | RCC_APB2ENR_AFIOEN | RCC_APB2ENR_ADC1EN | RCC_APB2ENR_USART1EN; // ADC时钟配置 RCC->CFGR &= ~RCC_CFGR_ADCPRE; RCC->CFGR |= RCC_CFGR_ADCPRE_DIV6; } // GPIO配置 void GPIO_Configuration(void) { // PA0 - 光敏电阻(模拟输入) GPIOA->CRL &= ~(GPIO_CRL_MODE0 | GPIO_CRL_CNF0); GPIOA->CRL |= GPIO_CRL_CNF0_1; // 模拟输入模式 // PC13 - 继电器控制(推挽输出) GPIOC->CRH &= ~(GPIO_CRH_MODE13 | GPIO_CRH_CNF13); GPIOC->CRH |= GPIO_CRH_MODE13_1; // 输出模式,最大速度2MHz // PA9 - USART1 TX (推挽复用输出) GPIOA->CRH &= ~(GPIO_CRH_MODE9 | GPIO_CRH_CNF9); GPIOA->CRH |= GPIO_CRH_MODE9_1 | GPIO_CRH_CNF9_1; // PA10 - USART1 RX (浮空输入) GPIOA->CRH &= ~(GPIO_CRH_MODE10 | GPIO_CRH_CNF10); GPIOA->CRH |= GPIO_CRH_CNF10_0; } // ADC配置 void ADC_Configuration(void) { // 复位ADC1 ADC1->CR2 = 0; // 配置ADC1 ADC1->CR1 = 0; ADC1->CR2 = ADC_CR2_ADON | ADC_CR2_EXTSEL | ADC_CR2_EXTTRIG; // 采样时间配置 ADC1->SMPR2 &= ~ADC_SMPR2_SMP0; ADC1->SMPR2 |= ADC_SMPR2_SMP0_2 | ADC_SMPR2_SMP0_1; // 239.5周期 // 规则序列配置 ADC1->SQR1 = 0; ADC1->SQR2 = 0; ADC1->SQR3 = LIGHT_SENSOR_ADC_CHANNEL; // 校准ADC ADC1->CR2 |= ADC_CR2_RSTCAL; while(ADC1->CR2 & ADC_CR2_RSTCAL); ADC1->CR2 |= ADC_CR2_CAL; while(ADC1->CR2 & ADC_CR2_CAL); } // USART配置 void USART_Configuration(void) { // 波特率配置 115200 USART1->BRR = 0x0271; // 使能USART, 发送, 接收, 接收中断 USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE | USART_CR1_RXNEIE; } // NVIC配置 void NVIC_Configuration(void) { NVIC->ISER[0] |= (1 << USART1_IRQn); // 使能USART1中断 } // 读取ADC值 uint16_t Read_ADC(uint8_t channel) { // 设置通道 ADC1->SQR3 = channel; // 启动转换 ADC1->CR2 |= ADC_CR2_ADON; Delay_ms(1); ADC1->CR2 |= ADC_CR2_ADON; // 等待转换完成 while(!(ADC1->SR & ADC_SR_EOC)); return ADC1->DR; } // 延时函数 void Delay_ms(uint32_t nTime) { volatile uint32_t i, j; for(i = 0; i < nTime; i++) for(j = 0; j < 8000; j++); } // 继电器控制 void Control_Relay(uint8_t state) { if(state) GPIOC->BSRR = RELAY_PIN; // 打开继电器 else GPIOC->BRR = RELAY_PIN; // 关闭继电器 light_status = state; } // 读取光照强度 uint16_t Read_Light_Sensor(void) { uint16_t adc_value = Read_ADC(LIGHT_SENSOR_ADC_CHANNEL); // 转换为光照强度值 (0-1000) return (adc_value * 1000) / 4095; } // 发送数据到Zigbee void Send_To_Zigbee(uint8_t* data, uint8_t len) { for(uint8_t i = 0; i < len; i++) { while(!(USART1->SR & USART_SR_TXE)); USART1->DR = data[i]; } } // 自动光照控制 void Auto_Light_Control(void) { if(auto_mode) { if(light_level < 300) // 光照阈值 Control_Relay(1); // 打开灯光 else Control_Relay(0); // 关闭灯光 } } // 处理Zigbee命令 void Process_Zigbee_Command(uint8_t* cmd) { switch(cmd[0]) { case 'O': // ON auto_mode = 0; Control_Relay(1); break; case 'F': // OFF auto_mode = 0; Control_Relay(0); break; case 'A': // AUTO auto_mode = 1; break; case 'G': // GET_STATUS { uint8_t status_msg[4]; status_msg[0] = 'S'; status_msg[1] = light_status ? '1' : '0'; status_msg[2] = (light_level / 100) + '0'; status_msg[3] = (light_level % 100 / 10) + '0'; Send_To_Zigbee(status_msg, 4); } break; } } // USART1中断服务函数 void USART1_IRQHandler(void) { if(USART1->SR & USART_SR_RXNE) { uint8_t received_data = USART1->DR; Process_Zigbee_Command(&received_data); } } // 主函数 int main(void) { // 系统初始化 RCC_Configuration(); GPIO_Configuration(); ADC_Configuration(); USART_Configuration(); NVIC_Configuration(); // 初始状态 Control_Relay(0); while(1) { // 读取光照强度 light_level = Read_Light_Sensor(); // 自动控制 Auto_Light_Control(); // 延时 Delay_ms(1000); } } 项目核心代码#include "stm32f10x.h" // 引脚定义 #define LIGHT_SENSOR_PIN GPIO_Pin_0 // PA0 - 光照传感器 #define RELAY_PIN GPIO_Pin_1 // PA1 - 继电器控制 #define ZIGBEE_TX_PIN GPIO_Pin_9 // PA9 - Zigbee TX #define ZIGBEE_RX_PIN GPIO_Pin_10 // PA10 - Zigbee RX // 光照阈值 #define LIGHT_THRESHOLD 500 // 设备状态 typedef enum { LIGHT_OFF = 0, LIGHT_ON = 1 } LightState; // 场景模式 typedef enum { MODE_NORMAL = 0, MODE_ALL_ON = 1, MODE_ALL_OFF = 2, MODE_NIGHT = 3 } SceneMode; // 全局变量 volatile LightState currentLightState = LIGHT_OFF; volatile SceneMode currentSceneMode = MODE_NORMAL; volatile uint16_t lightSensorValue = 0; // 函数声明 void System_Init(void); void GPIO_Init(void); void USART_Init(void); void ADC_Init(void); void TIM_Init(void); uint16_t Read_Light_Sensor(void); void Control_Light(LightState state); void Process_Zigbee_Command(uint8_t* command); void Send_Zigbee_Data(uint8_t* data, uint8_t len); void Process_Scene_Mode(SceneMode mode); int main(void) { System_Init(); while(1) { // 读取光照传感器值 lightSensorValue = Read_Light_Sensor(); // 自动模式下的光照控制 if(currentSceneMode == MODE_NORMAL) { if(lightSensorValue < LIGHT_THRESHOLD) { Control_Light(LIGHT_ON); } else { Control_Light(LIGHT_OFF); } } else if(currentSceneMode == MODE_NIGHT) { // 夜间模式:仅在非常暗时开启 if(lightSensorValue < (LIGHT_THRESHOLD / 2)) { Control_Light(LIGHT_ON); } else { Control_Light(LIGHT_OFF); } } // 处理Zigbee通信 if(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) != RESET) { uint8_t rxData[32]; uint8_t index = 0; // 读取Zigbee数据 while(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) != RESET && index < 32) { rxData[index++] = USART_ReceiveData(USART1); } // 处理命令 Process_Zigbee_Command(rxData); } // 定时上报状态 static uint32_t lastReportTime = 0; if(SystemCoreClock - lastReportTime > 5000000) // 5秒上报一次 { uint8_t statusData[4]; statusData[0] = 0xAA; // 帧头 statusData[1] = currentLightState; statusData[2] = lightSensorValue >> 8; statusData[3] = lightSensorValue & 0xFF; Send_Zigbee_Data(statusData, 4); lastReportTime = SystemCoreClock; } } } void System_Init(void) { // 开启时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_AFIOEN | RCC_APB2ENR_USART1EN | RCC_APB2ENR_ADC1EN; GPIO_Init(); USART_Init(); ADC_Init(); TIM_Init(); // 配置NVIC NVIC_EnableIRQ(USART1_IRQn); } void GPIO_Init(void) { // PA0 - 光照传感器输入 GPIOA->CRL &= ~(GPIO_CRL_MODE0 | GPIO_CRL_CNF0); GPIOA->CRL |= GPIO_CRL_CNF0_1; // 模拟输入 // PA1 - 继电器控制输出 GPIOA->CRL &= ~(GPIO_CRL_MODE1 | GPIO_CRL_CNF1); GPIOA->CRL |= GPIO_CRL_MODE1_0; // 推挽输出,2MHz // PA9 - USART1 TX 复用推挽输出 GPIOA->CRH &= ~(GPIO_CRH_MODE9 | GPIO_CRH_CNF9); GPIOA->CRH |= GPIO_CRH_MODE9_0 | GPIO_CRH_CNF9_1; // PA10 - USART1 RX 浮空输入 GPIOA->CRH &= ~(GPIO_CRH_MODE10 | GPIO_CRH_CNF10); GPIOA->CRH |= GPIO_CRH_CNF10_0; } void USART_Init(void) { // 波特率115200 USART1->BRR = 72000000 / 115200; // 使能USART, RXNE中断, 收发使能 USART1->CR1 = USART_CR1_UE | USART_CR1_RXNEIE | USART_CR1_TE | USART_CR1_RE; } void ADC_Init(void) { // ADC1初始化 ADC1->CR2 = ADC_CR2_ADON; // 开启ADC ADC1->SQR3 = 0; // 通道0 ADC1->SQR1 = 0; // 1个转换 } void TIM_Init(void) { // 定时器用于延时和定时任务 RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; TIM2->PSC = 7200 - 1; // 10kHz TIM2->ARR = 10000 - 1; // 1秒 TIM2->CR1 = TIM_CR1_CEN; } uint16_t Read_Light_Sensor(void) { ADC1->CR2 |= ADC_CR2_ADON; // 启动转换 ADC1->CR2 |= ADC_CR2_SWSTART; // 等待转换完成 while(!(ADC1->SR & ADC_SR_EOC)); return ADC1->DR; } void Control_Light(LightState state) { if(state == LIGHT_ON) { GPIOA->BSRR = RELAY_PIN; // 置位,开灯 currentLightState = LIGHT_ON; } else { GPIOA->BRR = RELAY_PIN; // 复位,关灯 currentLightState = LIGHT_OFF; } } void Process_Zigbee_Command(uint8_t* command) { if(command[0] == 0x55) // 命令帧头 { switch(command[1]) { case 0x01: // 开关控制 Control_Light((LightState)command[2]); break; case 0x02: // 场景模式 Process_Scene_Mode((SceneMode)command[2]); break; case 0x03: // 状态查询 { uint8_t status[3]; status[0] = 0xBB; status[1] = currentLightState; status[2] = currentSceneMode; Send_Zigbee_Data(status, 3); } break; } } } void Send_Zigbee_Data(uint8_t* data, uint8_t len) { for(uint8_t i = 0; i < len; i++) { while(!(USART1->SR & USART_SR_TXE)); USART1->DR = data[i]; } } void Process_Scene_Mode(SceneMode mode) { currentSceneMode = mode; switch(mode) { case MODE_ALL_ON: Control_Light(LIGHT_ON); break; case MODE_ALL_OFF: Control_Light(LIGHT_OFF); break; case MODE_NIGHT: // 夜间模式逻辑在main循环中处理 break; case MODE_NORMAL: // 正常模式逻辑在main循环中处理 break; } } // USART1中断服务函数 void USART1_IRQHandler(void) { if(USART1->SR & USART_SR_RXNE) { // RXNE标志会在读取DR时自动清除 volatile uint8_t temp = USART1->DR; } } 总结该系统基于Zigbee无线通信技术,成功构建了一个高效、灵活的智能家居灯光控制系统。通过协调器、路由器和终端设备的协同工作,系统实现了稳定的网络组建,支持环境光照强度的实时监测与灯具的智能控制,为用户提供便捷的远程操作体验。硬件方面,系统采用STM32单片机与Zigbee模块(如CC2530)作为核心控制单元,结合光敏电阻模块进行光照感知,继电器模块执行灯光开关操作,并通过Zigbee-USB协调器连接上位机或网关,确保手机APP能够远程控制任意房间的灯光。整体而言,该系统不仅支持多种场景模式,如“全开”、“全关”和“夜间模式”,还提升了家居自动化的智能化水平和能源效率,适用于现代家庭和商业环境,充分体现了Zigbee技术在智能家居领域的应用优势。
  • [技术干货] 基于STM32与AI的导盲杖系统
    项目开发背景视障人士在日常出行中面临着巨大的挑战,他们依赖传统的导盲杖或导盲犬来辅助导航,但这些方法存在诸多局限性。传统导盲杖只能探测到地面附近的障碍物,无法识别高处的物体或动态移动的目标,如行人或车辆,这容易导致碰撞风险。此外,在光线不足的环境中,视障人士的移动安全进一步受到威胁,现有的设备往往缺乏智能化的适应能力,无法提供实时、准确的障碍物信息反馈。随着嵌入式系统和人工智能技术的快速发展,智能辅助设备为解决上述问题提供了新的可能性。微控制器如STM32能够高效处理传感器数据,而AI模块如K210则具备强大的图像识别能力,可以快速检测和分类多种障碍物。这些技术的结合,使得开发一种多功能导盲杖系统成为现实,它不仅能够增强障碍物探测的精度,还能通过智能反馈机制提升用户的安全性和独立性。本项目旨在设计一种基于STM32与AI的智能导盲杖系统,通过集成超声波测距和K210目标检测功能,实现对前方及地面障碍物的全面监控。系统通过振动和语音提示向用户传达障碍物类型和距离信息,并在昏暗环境下自动启动照明,从而弥补传统导盲工具的不足。这一创新应用不仅提升了视障人士的出行体验,还推动了辅助技术向更智能化、人性化的方向发展。设计实现的功能(1)使用超声波模块实时检测前方及地面障碍物距离。(2)集成K210 AI模块,具备目标检测功能,可识别行人、车辆等特定障碍物。(3)通过不同模式的振动和语音提示,向使用者反馈障碍物类型和距离。(4)在光线昏暗环境自动点亮LED灯带进行照明。项目硬件模块组成(1) 主控芯片:STM32F103C8T6单片机。(2) AI核心:K210开发板(负责运行YOLO快速目标检测模型)。(3) 测距模块:HC-SR04超声波传感器(多个)。(4) 反馈模块:振动电机、SYN6288语音播报模块。(5) 供电模块:18650锂电池组与充放电管理模块。设计意义该导盲杖系统通过集成超声波传感器与AI目标检测技术,能够实时感知前方及地面障碍物,有效帮助视障人士规避碰撞风险,显著提升户外行走的安全性。系统利用K210模块运行YOLO模型识别行人、车辆等动态障碍物,弥补了传统导盲工具仅依赖距离检测的不足,使使用者能更全面地感知复杂环境中的潜在危险。通过振动与语音的多模式反馈机制,系统可根据障碍物类型和距离提供差异化提示,增强信息传递的直观性和及时性,帮助使用者快速做出反应。在光线昏暗环境下,自动点亮LED灯带的功能不仅改善了使用者的可视条件,还增加了夜间行动的独立性。硬件设计以STM32为核心协调各模块,结合高效供电方案,确保了系统的稳定性和便携性。整体设计体现了科技赋能残障辅助设备的实用价值,为视障人群提供了更智能、可靠的出行支持。设计思路该系统以STM32F103C8T6单片机作为主控核心,负责协调和管理各硬件模块的运作。通过实时采集超声波传感器数据,并结合K210 AI模块的处理结果,实现障碍物的精确检测与分类,最终通过振动和语音反馈为用户提供导航辅助。超声波模块采用多个HC-SR04传感器,分别部署于导盲杖的前方和底部,用于持续监测前方障碍物及地面凹陷或台阶的距离。STM32通过定时触发超声波脉冲并计算回波时间,转换为实际距离数据,当检测到障碍物距离低于设定阈值时,立即启动相应处理流程。K210开发板作为AI处理单元,运行优化的YOLO目标检测模型,能够实时识别行人、车辆等特定障碍物类型。STM32通过串口通信与K210交互,获取识别结果和位置信息,从而区分普通物体与动态障碍物,提升系统的环境感知能力。反馈机制通过振动电机和SYN6288语音模块实现,STM32根据障碍物类型和距离控制振动模式(如连续振动或间歇振动),同时驱动语音模块播报具体警告信息,例如“前方行人”或“地面障碍”,确保使用者能够直观理解周围环境状况。照明系统通过光敏传感器检测环境光线强度,当光线低于预设值时,STM32自动控制LED灯带点亮,为昏暗环境提供辅助照明,增强用户的安全性和使用便利性。整个系统由18650锂电池组供电,配合充放电管理模块确保稳定运行。框架图+-----------------------------+ | 供电模块 | | (18650锂电池组 + | | 充放电管理模块) | +-----------------------------+ | | 供电 v +-----------------------------+ | STM32F103C8T6 | | (主控制器) | +-----------------------------+ | | | | 测距数据 | AI检测结果 | 控制信号 v v v +----------+ +----------+ +-------------------+ | HC-SR04 | | K210 | | 反馈与照明模块 | | 超声波 | | 开发板 | | - 振动电机 | | 传感器 | | (YOLO | | - SYN6288语音模块 | | (多个) | | 目标检测)| | - LED灯带 | +----------+ +----------+ +-------------------+ | | 图像输入 v +-----------+ | 摄像头 | | (视觉采集)| +-----------+ 系统总体设计该系统总体设计基于STM32微控制器与K210 AI模块构建一个智能导盲杖,旨在通过多传感器融合和智能反馈机制辅助视障人士安全导航。系统以STM32F103C8T6作为主控核心,负责协调各硬件模块的数据处理与控制逻辑,确保实时响应和高效运行。在环境感知方面,系统部署多个HC-SR04超声波传感器,用于持续检测前方及地面障碍物的距离信息,同时集成K210开发板运行YOLO目标检测算法,能够快速识别行人、车辆等特定障碍物类型,并将检测数据传送至STM32进行综合分析。反馈机制通过振动电机和SYN6288语音播报模块实现,STM32根据障碍物类型和距离生成不同的振动模式,并触发语音提示,向使用者清晰传达环境信息。此外,系统具备自动照明功能,在检测到光线昏暗时控制LED灯带点亮,以提升使用者的可视性。供电部分采用18650锂电池组结合充放电管理模块,为整个系统提供稳定可靠的电源支持,确保设备在户外环境中长时间工作。所有硬件模块通过紧凑的电路设计集成于导盲杖结构中,实现功能与便携性的平衡。系统功能总结功能名称功能描述超声波障碍物检测使用超声波模块实时检测前方及地面障碍物距离AI目标检测集成K210 AI模块,具备目标检测功能,可识别行人、车辆等特定障碍物振动与语音反馈通过不同模式的振动和语音提示,向使用者反馈障碍物类型和距离自动照明在光线昏暗环境自动点亮LED灯带进行照明设计的各个功能模块描述主控芯片模块采用STM32F103C8T6单片机作为系统的核心控制器,负责协调各个模块的运行,处理来自超声波传感器和K210模块的数据,并根据检测结果控制反馈装置和照明功能,确保系统整体稳定高效工作。AI核心模块使用K210开发板运行YOLO快速目标检测模型,能够实时识别前方环境中的特定障碍物,如行人和车辆,并将识别结果发送给主控芯片,以增强障碍物检测的准确性和智能化水平。测距模块通过多个HC-SR04超声波传感器实时检测前方及地面障碍物的距离,数据被传输至主控芯片进行分析,从而提供精确的距离信息用于后续反馈处理。反馈模块包括振动电机和SYN6288语音播报模块,主控芯片根据障碍物类型和距离信息控制振动电机产生不同模式的振动,同时驱动语音模块播报提示内容,帮助使用者及时了解周围环境状况。供电模块由18650锂电池组和充放电管理模块组成,为整个系统提供稳定的电源供应,确保各功能模块在长时间使用中可靠运行,同时支持充放电管理以延长电池寿命。照明模块通过LED灯带在光线昏暗环境下自动点亮,由主控芯片控制实现智能照明功能,为使用者提供额外的环境可见度支持。上位机代码设计#include <iostream> #include <string> #include <vector> #include <windows.h> #include <thread> #include <chrono> #include <map> // 串口通信类 class SerialPort { private: HANDLE hSerial; bool connected; COMSTAT status; DWORD errors; public: SerialPort(const char* portName) : connected(false) { hSerial = CreateFileA(portName, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (hSerial == INVALID_HANDLE_VALUE) { std::cerr << "错误:无法打开串口 " << portName << std::endl; return; } DCB dcbSerialParams = {0}; dcbSerialParams.DCBlength = sizeof(dcbSerialParams); if (!GetCommState(hSerial, &dcbSerialParams)) { std::cerr << "错误:无法获取串口状态" << std::endl; CloseHandle(hSerial); return; } dcbSerialParams.BaudRate = CBR_115200; dcbSerialParams.ByteSize = 8; dcbSerialParams.StopBits = ONESTOPBIT; dcbSerialParams.Parity = NOPARITY; if (!SetCommState(hSerial, &dcbSerialParams)) { std::cerr << "错误:无法设置串口参数" << std::endl; CloseHandle(hSerial); return; } PurgeComm(hSerial, PURGE_RXCLEAR | PURGE_TXCLEAR); connected = true; std::cout << "成功连接到串口 " << portName << std::endl; } ~SerialPort() { if (connected) { CloseHandle(hSerial); } } bool isConnected() const { return connected; } int writeData(const char* buffer, unsigned int buf_size) { DWORD bytesSend; if (!WriteFile(hSerial, buffer, buf_size, &bytesSend, 0)) { std::cerr << "错误:串口写入失败" << std::endl; return -1; } return bytesSend; } std::string readLine() { if (!connected) return ""; DWORD bytesRead; char buffer[256]; std::string result; while (true) { if (ReadFile(hSerial, buffer, 1, &bytesRead, NULL) && bytesRead > 0) { if (buffer[0] == '\n') { break; } result += buffer[0]; } else { break; } } return result; } }; // 数据解析类 class DataParser { private: std::map<std::string, std::string> obstacleData; public: void parse(const std::string& data) { obstacleData.clear(); size_t pos = 0; std::string token; std::string s = data; // 解析格式:TYPE:VALUE|TYPE:VALUE while ((pos = s.find('|')) != std::string::npos) { token = s.substr(0, pos); size_t colonPos = token.find(':'); if (colonPos != std::string::npos) { std::string key = token.substr(0, colonPos); std::string value = token.substr(colonPos + 1); obstacleData[key] = value; } s.erase(0, pos + 1); } // 处理最后一个字段 size_t colonPos = s.find(':'); if (colonPos != std::string::npos) { std::string key = s.substr(0, colonPos); std::string value = s.substr(colonPos + 1); obstacleData[key] = value; } } void display() { std::cout << "\n=== 障碍物检测报告 ===" << std::endl; for (const auto& pair : obstacleData) { if (pair.first == "FRONT_DIST") { std::cout << "前方距离: " << pair.second << "cm" << std::endl; } else if (pair.first == "GROUND_DIST") { std::cout << "地面距离: " << pair.second << "cm" << std::endl; } else if (pair.first == "OBJECT") { std::cout << "检测对象: " << pair.second << std::endl; } else if (pair.first == "LIGHT") { std::cout << "光照状态: " << (pair.second == "1" ? "昏暗-已开启LED" : "正常") << std::endl; } else if (pair.first == "VIBRATION") { std::cout << "振动模式: " << pair.second << std::endl; } } std::cout << "=====================" << std::endl; } void generateAlert() { if (obstacleData.find("FRONT_DIST") != obstacleData.end()) { int dist = std::stoi(obstacleData["FRONT_DIST"]); if (dist < 50) { std::cout << "!!! 警告:前方 " << dist << "cm 处有障碍物!" << std::endl; } } if (obstacleData.find("OBJECT") != obstacleData.end()) { std::string obj = obstacleData["OBJECT"]; if (obj != "NONE") { std::cout << "!!! AI识别警告:检测到 " << obj << "!" << std::endl; } } } }; // 主控制类 class BlindStickMonitor { private: SerialPort serial; DataParser parser; bool monitoring; public: BlindStickMonitor(const std::string& port) : serial(port.c_str()), monitoring(false) {} void startMonitoring() { if (!serial.isConnected()) { std::cerr << "错误:串口未连接,无法启动监控" << std::endl; return; } monitoring = true; std::cout << "开始监控导盲杖系统..." << std::endl; std::cout << "按Ctrl+C停止监控" << std::endl; while (monitoring) { std::string data = serial.readLine(); if (!data.empty()) { std::cout << "\n收到原始数据: " << data << std::endl; parser.parse(data); parser.display(); parser.generateAlert(); } std::this_thread::sleep_for(std::chrono::milliseconds(500)); } } void stopMonitoring() { monitoring = false; std::cout << "停止监控" << std::endl; } void sendCommand(const std::string& cmd) { if (serial.isConnected()) { std::string fullCmd = cmd + "\n"; serial.writeData(fullCmd.c_str(), fullCmd.length()); std::cout << "发送命令: " << cmd << std::endl; } } }; // 控制台界面 void displayMenu() { std::cout << "\n===== 导盲杖监控系统 =====" << std::endl; std::cout << "1. 开始监控" << std::endl; std::cout << "2. 发送测试命令" << std::endl; std::cout << "3. 设置串口" << std::endl; std::cout << "4. 退出" << std::endl; std::cout << "请选择操作: "; } int main() { std::string port = "COM3"; BlindStickMonitor monitor(port); int choice = 0; while (choice != 4) { displayMenu(); std::cin >> choice; switch (choice) { case 1: monitor.startMonitoring(); break; case 2: monitor.sendCommand("TEST"); break; case 3: std::cout << "请输入串口名称 (如 COM3): "; std::cin >> port; break; case 4: std::cout << "退出系统" << std::endl; break; default: std::cout << "无效选择" << std::endl; } } return 0; } 模块代码设计#include "stm32f10x.h" // 超声波引脚定义 #define TRIG_PIN GPIO_Pin_1 #define ECHO_PIN GPIO_Pin_0 #define TRIG_PORT GPIOA #define ECHO_PORT GPIOA // 振动电机引脚定义 #define VIBRATE_PIN GPIO_Pin_2 #define VIBRATE_PORT GPIOA // LED灯带引脚定义 #define LED_PIN GPIO_Pin_3 #define LED_PORT GPIOA // 光敏电阻ADC通道 #define LIGHT_SENSOR_ADC_CHANNEL ADC_Channel_4 // 系统时钟初始化 void RCC_Configuration(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_AFIOEN | RCC_APB2ENR_ADC1EN; RCC->APB1ENR |= RCC_APB1ENR_TIM2EN | RCC_APB1ENR_USART2EN; } // GPIO初始化 void GPIO_Configuration(void) { // 超声波TRIG引脚推挽输出 GPIOA->CRL &= 0xFFFFFF0F; GPIOA->CRL |= 0x00000030; // 超声波ECHO引脚浮空输入 GPIOA->CRL &= 0xFFFFFFF0; GPIOA->CRL |= 0x00000040; // 振动电机推挽输出 GPIOA->CRL &= 0xFFFFF0FF; GPIOA->CRL |= 0x00000300; // LED灯带推挽输出 GPIOA->CRL &= 0xFFFF0FFF; GPIOA->CRL |= 0x00003000; // USART2引脚配置 GPIOA->CRL &= 0xFFFF00FF; GPIOA->CRL |= 0x00008B00; // PA2复用推挽输出(TX), PA3浮空输入(RX) } // 定时器2初始化(用于超声波测距) void TIM2_Configuration(void) { TIM2->CR1 = 0x0000; TIM2->CR2 = 0x0000; TIM2->PSC = 71; // 72MHz/72 = 1MHz TIM2->ARR = 0xFFFF; TIM2->CNT = 0; TIM2->CR1 |= TIM_CR1_CEN; } // ADC初始化(用于光敏电阻) void ADC1_Configuration(void) { ADC1->CR2 = ADC_CR2_ADON | ADC_CR2_CAL; while(ADC1->CR2 & ADC_CR2_CAL); ADC1->SMPR2 |= ADC_SMPR2_SMP4; // 通道4采样时间239.5周期 ADC1->CR2 |= ADC_CR2_CONT; ADC1->CR2 |= ADC_CR2_ADON; } // USART2初始化(与K210通信) void USART2_Configuration(void) { USART2->BRR = 0x1D4C; // 115200 @72MHz USART2->CR1 = USART_CR1_TE | USART_CR1_RE | USART_CR1_UE; USART2->CR1 |= USART_CR1_RXNEIE; NVIC->ISER[0] |= (1 << (USART2_IRQn & 0x1F)); } // 发送超声波触发信号 void Ultrasonic_Trigger(void) { TRIG_PORT->BSRR = TRIG_PIN; for(volatile int i = 0; i < 100; i++); TRIG_PORT->BRR = TRIG_PIN; } // 获取超声波距离(厘米) uint16_t Ultrasonic_GetDistance(void) { Ultrasonic_Trigger(); TIM2->CNT = 0; while(!(ECHO_PORT->IDR & ECHO_PIN) && TIM2->CNT < 1000); if(TIM2->CNT >= 1000) return 0; uint32_t start_time = TIM2->CNT; while((ECHO_PORT->IDR & ECHO_PIN) && TIM2->CNT < 30000); uint32_t end_time = TIM2->CNT; if(end_time >= 30000) return 0; uint32_t pulse_width = end_time - start_time; return (uint16_t)(pulse_width / 58); // 转换为厘米 } // 读取光照强度 uint16_t LightSensor_Read(void) { ADC1->SQR3 = LIGHT_SENSOR_ADC_CHANNEL; ADC1->CR2 |= ADC_CR2_SWSTART; while(!(ADC1->SR & ADC_SR_EOC)); return ADC1->DR; } // 控制振动电机 void Vibrate_Control(uint8_t mode, uint8_t intensity) { switch(mode) { case 0: // 连续振动 VIBRATE_PORT->BSRR = VIBRATE_PIN; break; case 1: // 间断振动 if(TIM2->CNT % 1000 < intensity * 10) VIBRATE_PORT->BSRR = VIBRATE_PIN; else VIBRATE_PORT->BRR = VIBRATE_PIN; break; default: VIBRATE_PORT->BRR = VIBRATE_PIN; break; } } // 控制LED灯带 void LED_Control(uint8_t state) { if(state) LED_PORT->BSRR = LED_PIN; else LED_PORT->BRR = LED_PIN; } // USART2发送数据到K210 void USART2_SendData(uint8_t data) { while(!(USART2->SR & USART_SR_TXE)); USART2->DR = data; } // USART2发送字符串 void USART2_SendString(char *str) { while(*str) { USART2_SendData(*str++); } } // USART2中断服务函数 void USART2_IRQHandler(void) { if(USART2->SR & USART_SR_RXNE) { uint8_t received_data = USART2->DR; // 处理从K210接收到的AI识别结果 // 这里可以根据接收到的数据控制振动和语音提示 } } // 系统主循环 int main(void) { RCC_Configuration(); GPIO_Configuration(); TIM2_Configuration(); ADC1_Configuration(); USART2_Configuration(); uint16_t distance = 0; uint16_t light_level = 0; while(1) { // 超声波测距 distance = Ultrasonic_GetDistance(); // 光照检测 light_level = LightSensor_Read(); // 根据光照自动控制LED if(light_level < 1000) // 光线较暗 { LED_Control(1); } else { LED_Control(0); } // 根据距离控制振动模式 if(distance > 0 && distance < 50) { if(distance < 20) Vibrate_Control(0, 100); // 近距离连续振动 else Vibrate_Control(1, 80); // 中距离间断振动 // 发送距离信息到K210进行AI处理 USART2_SendString("DIST:"); USART2_SendData((distance >> 8) & 0xFF); USART2_SendData(distance & 0xFF); } else { Vibrate_Control(0, 0); // 停止振动 } // 延时 for(volatile int i = 0; i < 100000; i++); } } 项目核心代码#include "stm32f10x.h" // 模块初始化函数声明 void Ultrasonic_Init(void); void K210_Init(void); void Vibration_Init(void); void Voice_Init(void); void LED_Init(void); void Timer_Init(void); // 功能函数声明 uint32_t Get_Ultrasonic_Distance(uint8_t sensor_id); uint8_t Get_K210_Detection(void); void Set_Vibration_Mode(uint8_t mode); void Voice_Alert(uint8_t alert_type, uint32_t distance); void LED_Control(uint8_t state); uint8_t Check_Light_Intensity(void); // 全局变量 volatile uint32_t system_tick = 0; uint8_t obstacle_type = 0; uint32_t front_distance = 0; uint32_t ground_distance = 0; int main(void) { // 系统初始化 SystemInit(); // 外设初始化 Ultrasonic_Init(); K210_Init(); Vibration_Init(); Voice_Init(); LED_Init(); Timer_Init(); // 启动定时器 TIM_Cmd(TIM2, ENABLE); while(1) { // 1. 超声波测距 front_distance = Get_Ultrasonic_Distance(0); // 前方传感器 ground_distance = Get_Ultrasonic_Distance(1); // 地面传感器 // 2. AI目标检测 obstacle_type = Get_K210_Detection(); // 3. 障碍物处理逻辑 if(front_distance < 200 || ground_distance < 150) // 距离单位:cm { // 根据距离设置振动模式 if(front_distance < 100 || ground_distance < 80) { Set_Vibration_Mode(2); // 强烈振动 } else { Set_Vibration_Mode(1); // 轻微振动 } // 语音提示 Voice_Alert(obstacle_type, front_distance); } else { Set_Vibration_Mode(0); // 停止振动 } // 4. 光线检测与LED控制 if(Check_Light_Intensity() == 1) { LED_Control(1); // 开启LED照明 } else { LED_Control(0); // 关闭LED照明 } // 系统延时 Delay_ms(100); } } // 定时器中断服务函数 void TIM2_IRQHandler(void) { if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) { system_tick++; TIM_ClearITPendingBit(TIM2, TIM_IT_Update); } } // 超声波初始化 void Ultrasonic_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN; // 开启GPIOA时钟 // 配置Trig引脚(PA1)为推挽输出 GPIOA->CRL &= 0xFFFFF0FF; GPIOA->CRL |= 0x00000300; // 配置Echo引脚(PA2)为浮空输入 GPIOA->CRL &= 0xFFFFF0FF; GPIOA->CRL |= 0x00000400; } // K210通信初始化 void K210_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN; // 开启GPIOA时钟 RCC->APB2ENR |= RCC_APB2ENR_USART1EN; // 开启USART1时钟 // 配置PA9(TX), PA10(RX)为复用推挽输出 GPIOA->CRH &= 0xFFFFF00F; GPIOA->CRH |= 0x000004B0; // USART1配置 USART1->BRR = 0x1D4C; // 115200 @72MHz USART1->CR1 = USART_CR1_TE | USART_CR1_RE | USART_CR1_UE; } // 振动电机初始化 void Vibration_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPBEN; // 开启GPIOB时钟 // 配置振动电机引脚(PB0,PB1)为推挽输出 GPIOB->CRL &= 0xFFFFFF00; GPIOB->CRL |= 0x00000033; } // 语音模块初始化 void Voice_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN; RCC->APB2ENR |= RCC_APB2ENR_USART2EN; // 配置PA2(TX), PA3(RX) GPIOA->CRL &= 0xFFFF00FF; GPIOA->CRL |= 0x00008B00; // USART2配置 USART2->BRR = 0x1D4C; // 115200 USART2->CR1 = USART_CR1_TE | USART_CR1_UE; } // LED初始化 void LED_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPCEN; // 配置LED引脚(PC13)为推挽输出 GPIOC->CRH &= 0xFF0FFFFF; GPIOC->CRH |= 0x00300000; } // 定时器初始化 void Timer_Init(void) { RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; TIM2->PSC = 7200 - 1; // 10kHz TIM2->ARR = 10000 - 1; // 1s中断 TIM2->DIER |= TIM_DIER_UIE; TIM2->CR1 |= TIM_CR1_CEN; NVIC_EnableIRQ(TIM2_IRQn); NVIC_SetPriority(TIM2_IRQn, 0); } // 简易延时函数 void Delay_ms(uint32_t ms) { uint32_t start_tick = system_tick; while((system_tick - start_tick) < ms); } // 获取超声波距离 uint32_t Get_Ultrasonic_Distance(uint8_t sensor_id) { uint32_t distance = 0; // 超声波测距实现代码 return distance; } // 获取K210检测结果 uint8_t Get_K210_Detection(void) { uint8_t result = 0; // K210通信代码 return result; } // 设置振动模式 void Set_Vibration_Mode(uint8_t mode) { switch(mode) { case 0: // 停止 GPIOB->BRR = GPIO_Pin_0 | GPIO_Pin_1; break; case 1: // 轻微振动 GPIOB->BSRR = GPIO_Pin_0; GPIOB->BRR = GPIO_Pin_1; break; case 2: // 强烈振动 GPIOB->BSRR = GPIO_Pin_0 | GPIO_Pin_1; break; } } // 语音报警 void Voice_Alert(uint8_t alert_type, uint32_t distance) { // 语音提示实现 } // LED控制 void LED_Control(uint8_t state) { if(state) GPIOC->BSRR = GPIO_Pin_13; else GPIOC->BRR = GPIO_Pin_13; } // 光线强度检测 uint8_t Check_Light_Intensity(void) { // 光线检测实现 return 0; } 总结该系统基于STM32微控制器和人工智能技术,构建了一款先进的智能导盲杖,旨在帮助视障人士安全导航。它通过超声波传感器实时检测前方及地面障碍物的距离,并结合K210 AI模块进行目标识别,可准确辨别行人、车辆等特定障碍物类型。系统采用多模式反馈机制,通过振动电机和语音播报模块向使用者传递障碍物的距离和类别信息,确保提示清晰直观。同时,在光线昏暗环境下,LED灯带会自动点亮以提供照明,增强使用安全性。硬件上,主控芯片采用STM32F103C8T6,AI处理由K210开发板运行YOLO模型负责,测距模块使用HC-SR04超声波传感器,反馈模块包括振动电机和SYN6288语音组件,供电则由18650锂电池组及管理模块支持。整体设计融合了传感与智能算法,为视障人士提供了高效、可靠的辅助工具。
  • [技术干货] 基于STM32的智能鱼塘监控系统
    项目开发背景随着水产养殖业的快速发展,鱼塘养殖作为重要的农业生产方式,在保障食品供应和促进农村经济中扮演着关键角色。然而,传统鱼塘管理主要依赖人工经验,养殖户需要频繁现场检测水质参数,如温度、pH值和溶解氧含量,这不仅效率低下,还容易因人为疏忽导致监测不及时,引发鱼类疾病或死亡,造成经济损失。在当前农业现代化趋势下,智能化和自动化技术正逐步应用于水产养殖领域,以提升管理效率和精准度。通过引入物联网和嵌入式系统,可以实现对鱼塘环境的实时监控与远程控制,帮助养殖户及时响应水质变化,降低运营成本,并促进可持续养殖模式的推广。基于此,本项目开发了一套基于STM32的智能鱼塘监控系统,旨在通过集成传感器、通信模块和执行设备,构建一个高效、可靠的解决方案。该系统能够自动采集和分析水质数据,支持用户远程监控和操作,从而有效解决传统养殖中的管理难题,推动鱼塘养殖向智能化、精细化方向发展。设计实现的功能(1) 实时监测鱼塘水体温度、pH值、溶解氧等关键水质参数。(2) 用户可通过手机APP远程查看水质数据,并手动控制增氧机、喂食机。(3) 系统可设置自动模式,根据水质参数阈值自动控制增氧机的启停。(4) 所有设备状态和水质历史数据可在APP端查询。项目硬件模块组成(1)主控芯片:STM32F103系列单片机(2)传感模块:DS18B20水温传感器、pH传感器、溶解氧传感器(3)通信模块:ESP8266 Wi-Fi模块(4)执行模块:继电器模块(控制增氧机、喂食机)(5)显示模块:LCD1602液晶显示屏(可选,用于本地显示)设计意义该系统通过集成传感器和无线通信技术,实现了对鱼塘水质的实时监测与设备远程控制,有效提升了水产养殖管理的自动化水平。用户无需频繁现场检查,即可通过手机APP获取关键数据并操作设备,显著减少了人力投入和操作时间。通过持续监控温度、pH值和溶解氧等参数,系统能够及时调整增氧机和喂食机的工作状态,确保鱼类生长环境稳定。这有助于预防水质恶化导致的疾病或死亡,从而提高养殖成功率和鱼类健康水平。系统的自动模式功能根据预设阈值智能控制设备运行,避免了人为疏忽可能引发的风险。同时,历史数据查询功能为用户提供了长期趋势分析的基础,支持更科学的养殖决策,优化资源利用。该设计推动了传统水产养殖向智能化转型,降低了能源和人力成本,提升了生产效益。它以实际应用为导向,为中小型养殖户提供了一种可靠、经济的解决方案,具有广泛的市场推广价值。设计思路该系统设计以STM32F103系列单片机为核心控制器,负责协调整个鱼塘监控系统的运行。系统通过集成多种传感器模块实时采集鱼塘水体的关键参数,包括DS18B20数字温度传感器测量水温、pH传感器检测酸碱度、溶解氧传感器监测水中氧气含量。这些传感器数据经过STM32的ADC或数字接口进行读取和处理,确保数据的准确性和实时性。数据处理后,STM32根据预设的阈值判断水质状态,并通过继电器模块控制增氧机和喂食机的启停。在自动模式下,系统会持续比较采集值与阈值,例如当溶解氧低于设定值时自动启动增氧机,实现智能化管理,减少人工干预。同时,继电器模块的设计允许安全驱动大功率设备,保障鱼塘设备的稳定运行。通信部分采用ESP8266 Wi-Fi模块,将STM32处理的数据上传到云端服务器或直接与手机APP交互。用户可以通过APP远程查看实时水质数据、手动控制设备开关,并查询历史记录,方便随时掌握鱼塘状况。Wi-Fi模块的配置确保数据传输的可靠性和低延迟,支持多种网络环境。用户交互方面,手机APP提供直观的界面显示水质参数和设备状态,并允许设置自动模式的阈值。此外,可选配的LCD1602液晶显示屏可用于本地实时显示关键数据,作为辅助监控手段,增强系统的实用性。整个设计注重稳定性和易用性,满足鱼塘日常管理的实际需求。框架图+----------------+ +-----------------+ +-------------------+ | 传感器模块 |------>| 主控芯片 |<----->| 通信模块 | | - DS18B20 | | STM32F103 | | ESP8266 Wi-Fi | | - pH传感器 | | | | | | - 溶解氧传感器 | +-----------------+ +-------------------+ +----------------+ | | | | v v +-----------------+ +-------------------+ | 执行模块 | | 手机APP | | - 继电器模块 | | (远程控制/查看) | | - 增氧机 | +-------------------+ | - 喂食机 | +-----------------+ | +-----------------+ | 显示模块 | | - LCD1602 | | (可选) | +-----------------+ 系统总体设计本系统基于STM32F103系列单片机作为核心控制器,旨在实现对鱼塘水质的智能化监控和管理。系统通过集成多种传感器模块,实时采集水体温度、pH值和溶解氧等关键参数,确保对鱼塘环境的全面监测。这些数据经过主控芯片处理和分析后,为后续的自动控制和远程交互提供基础。在数据采集方面,系统使用DS18B20水温传感器、pH传感器和溶解氧传感器分别获取相应的水质指标。STM32主控芯片定期读取传感器数据,并进行必要的校准和转换,以保证数据的准确性和可靠性。处理后的数据可用于本地显示或通过通信模块传输。远程通信和控制功能通过ESP8266 Wi-Fi模块实现,该模块将系统连接到互联网,使得用户能够通过手机APP实时查看水质参数。用户还可以通过APP手动发送指令,控制增氧机和喂食机的开关状态,实现远程操作。同时,系统支持自动模式,用户可在APP端设置水质参数的阈值,当监测数据超出范围时,系统自动通过继电器模块控制增氧机的启停,无需人工干预。对于本地显示,系统可选配LCD1602液晶显示屏,用于实时展示当前水质数据和设备运行状态,方便现场查看。所有设备状态和水质历史数据均被记录和存储,用户可通过手机APP查询历史记录,以便分析趋势和优化管理策略。整体设计注重实用性和稳定性,确保系统在鱼塘环境中高效运行。系统功能总结功能项功能描述实时监测实时监测鱼塘水体温度、pH值、溶解氧等关键水质参数。远程查看与控制用户可通过手机APP远程查看水质数据,并手动控制增氧机、喂食机。自动控制系统可设置自动模式,根据水质参数阈值自动控制增氧机的启停。数据查询所有设备状态和水质历史数据可在APP端查询。设计的各个功能模块描述主控模块基于STM32F103系列单片机,作为系统的核心处理单元,负责协调和控制所有其他模块的运行。它实时采集传感模块的数据,处理水质参数信息,并根据预设逻辑或用户指令控制执行模块的动作,同时管理通信模块的数据传输和显示模块的更新,确保系统稳定高效地工作。传感模块由DS18B20水温传感器、pH传感器和溶解氧传感器组成,用于持续监测鱼塘水体的关键水质指标。这些传感器将采集到的温度、pH值和溶解氧数据转换为电信号,传送给主控芯片进行分析和处理,以实现对鱼塘环境的实时监控。通信模块采用ESP8266 Wi-Fi模块,实现系统与互联网的无缝连接。通过该模块,主控芯片可以将水质数据和设备状态发送到云端服务器,用户即可通过手机APP远程查看实时信息、手动控制增氧机和喂食机,并查询历史记录,确保远程管理的便捷性和实时性。执行模块通过继电器模块控制增氧机和喂食机的操作,主控芯片根据用户通过APP发送的指令或自动模式下设定的水质参数阈值(如溶解氧水平)来驱动继电器,从而自动或手动启停这些设备,以维持鱼塘水质的稳定和鱼类的健康生长。显示模块可选配LCD1602液晶显示屏,用于在本地直观展示当前水质参数(如温度、pH值和溶解氧)以及设备运行状态。该模块为用户提供现场监控的便利,无需依赖远程设备即可快速获取系统信息,增强了系统的实用性和灵活性。上位机代码设计#include <iostream> #include <vector> #include <string> #include <thread> #include <chrono> #include <random> #include <ctime> #include <iomanip> #include <sstream> #include <map> #include <mutex> // 水质数据结构体 struct WaterQualityData { double temperature; // 温度 ℃ double pH; // pH值 double dissolvedOxygen; // 溶解氧 mg/L std::string timestamp; // 时间戳 bool aeratorStatus; // 增氧机状态 bool feederStatus; // 喂食机状态 }; // 设备控制命令 enum class DeviceCommand { AERATOR_ON, AERATOR_OFF, FEEDER_ON, FEEDER_OFF, GET_DATA }; // 智能鱼塘监控系统上位机类 class FishPondMonitor { private: std::vector<WaterQualityData> historyData; std::mutex dataMutex; bool isRunning; bool autoMode; double tempThreshold; double pHThreshold; double oxygenThreshold; // 模拟传感器数据生成 WaterQualityData generateSensorData() { std::random_device rd; std::mt19937 gen(rd()); std::uniform_real_distribution<> tempDist(15.0, 30.0); std::uniform_real_distribution<> pHDist(6.5, 8.5); std::uniform_real_distribution<> oxygenDist(5.0, 12.0); WaterQualityData data; data.temperature = tempDist(gen); data.pH = pHDist(gen); data.dissolvedOxygen = oxygenDist(gen); // 生成时间戳 auto now = std::chrono::system_clock::now(); auto time_t = std::chrono::system_clock::to_time_t(now); std::stringstream ss; ss << std::put_time(std::localtime(&time_t), "%Y-%m-%d %H:%M:%S"); data.timestamp = ss.str(); return data; } // 模拟与STM32通信 bool sendCommandToSTM32(DeviceCommand command) { std::this_thread::sleep_for(std::chrono::milliseconds(100)); return true; } WaterQualityData receiveDataFromSTM32() { std::this_thread::sleep_for(std::chrono::milliseconds(200)); return generateSensorData(); } public: FishPondMonitor() : isRunning(false), autoMode(true), tempThreshold(25.0), pHThreshold(7.0), oxygenThreshold(6.0) {} // 启动监控系统 void startMonitoring() { isRunning = true; std::cout << "智能鱼塘监控系统启动..." << std::endl; // 数据采集线程 std::thread dataThread([this]() { while (isRunning) { WaterQualityData newData = receiveDataFromSTM32(); std::lock_guard<std::mutex> lock(dataMutex); historyData.push_back(newData); // 保持历史数据在100条以内 if (historyData.size() > 100) { historyData.erase(historyData.begin()); } // 自动模式控制 if (autoMode) { if (newData.dissolvedOxygen < oxygenThreshold) { sendCommandToSTM32(DeviceCommand::AERATOR_ON); std::cout << "溶解氧过低,自动开启增氧机" << std::endl; } else if (newData.dissolvedOxygen > oxygenThreshold + 2.0) { sendCommandToSTM32(DeviceCommand::AERATOR_OFF); std::cout << "溶解氧正常,自动关闭增氧机" << std::endl; } } std::this_thread::sleep_for(std::chrono::seconds(5)); } }); dataThread.detach(); } // 停止监控系统 void stopMonitoring() { isRunning = false; std::cout << "监控系统停止" << std::endl; } // 获取当前水质数据 WaterQualityData getCurrentData() { std::lock_guard<std::mutex> lock(dataMutex); if (historyData.empty()) { return generateSensorData(); } return historyData.back(); } // 获取历史数据 std::vector<WaterQualityData> getHistoryData(int count = 10) { std::lock_guard<std::mutex> lock(dataMutex); if (historyData.size() <= count) { return historyData; } return std::vector<WaterQualityData>( historyData.end() - count, historyData.end()); } // 手动控制增氧机 bool controlAerator(bool turnOn) { DeviceCommand cmd = turnOn ? DeviceCommand::AERATOR_ON : DeviceCommand::AERATOR_OFF; bool success = sendCommandToSTM32(cmd); if (success) { std::cout << "增氧机" << (turnOn ? "开启" : "关闭") << std::endl; } return success; } // 手动控制喂食机 bool controlFeeder(bool turnOn) { DeviceCommand cmd = turnOn ? DeviceCommand::FEEDER_ON : DeviceCommand::FEEDER_OFF; bool success = sendCommandToSTM32(cmd); if (success) { std::cout << "喂食机" << (turnOn ? "开启" : "关闭") << std::endl; } return success; } // 设置自动模式 void setAutoMode(bool enabled) { autoMode = enabled; std::cout << "自动模式" << (enabled ? "开启" : "关闭") << std::endl; } // 设置阈值 void setThresholds(double temp, double pH, double oxygen) { tempThreshold = temp; pHThreshold = pH; oxygenThreshold = oxygen; std::cout << "阈值设置成功: 温度=" << temp << "℃, pH=" << pH << ", 溶解氧=" << oxygen << "mg/L" << std::endl; } // 显示当前状态 void displayCurrentStatus() { WaterQualityData current = getCurrentData(); std::cout << "\n=== 当前鱼塘状态 ===" << std::endl; std::cout << "时间: " << current.timestamp << std::endl; std::cout << "水温: " << current.temperature << " ℃" << std::endl; std::cout << "pH值: " << current.pH << std::endl; std::cout << "溶解氧: " << current.dissolvedOxygen << " mg/L" << std::endl; std::cout << "增氧机: " << (current.aeratorStatus ? "运行中" : "停止") << std::endl; std::cout << "喂食机: " << (current.feederStatus ? "运行中" : "停止") << std::endl; std::cout << "自动模式: " << (autoMode ? "开启" : "关闭") << std::endl; std::cout << "===================" << std::endl; } // 显示历史数据 void displayHistoryData(int count = 10) { std::vector<WaterQualityData> history = getHistoryData(count); std::cout << "\n=== 最近" << count << "条历史数据 ===" << std::endl; for (const auto& data : history) { std::cout << data.timestamp << " - " << "温度:" << data.temperature << "℃ " << "pH:" << data.pH << " " << "溶解氧:" << data.dissolvedOxygen << "mg/L" << std::endl; } std::cout << "===================" << std::endl; } }; // 手机APP模拟界面类 class MobileApp { private: FishPondMonitor& monitor; public: MobileApp(FishPondMonitor& mon) : monitor(mon) {} void showMainMenu() { int choice; do { std::cout << "\n===== 智能鱼塘监控APP =====" << std::endl; std::cout << "1. 查看当前状态" << std::endl; std::cout << "2. 查看历史数据" << std::endl; std::cout << "3. 手动控制增氧机" << std::endl; std::cout << "4. 手动控制喂食机" << std::endl; std::cout << "5. 设置自动模式" << std::endl; std::cout << "6. 设置报警阈值" << std::endl; std::cout << "0. 退出" << std::endl; std::cout << "请选择操作: "; std::cin >> choice; switch (choice) { case 1: monitor.displayCurrentStatus(); break; case 2: monitor.displayHistoryData(); break; case 3: controlAeratorMenu(); break; case 4: controlFeederMenu(); break; case 5: setAutoModeMenu(); break; case 6: setThresholdMenu(); break; case 0: std::cout << "退出APP" << std::endl; break; default: std::cout << "无效选择" << std::endl; } } while (choice != 0); } private: void controlAeratorMenu() { int choice; std::cout << "\n--- 增氧机控制 ---" << std::endl; std::cout << "1. 开启增氧机" << std::endl; std::cout << "2. 关闭增氧机" << std::endl; std::cout << "0. 返回" << std::endl; std::cout << "请选择: "; std::cin >> choice; if (choice == 1) { monitor.controlAerator(true); } else if (choice == 2) { monitor.controlAerator(false); } } void controlFeederMenu() { int choice; std::cout << "\n--- 喂食机控制 ---" << std::endl; std::cout << "1. 开启喂食机" << std::endl; std::cout << "2. 关闭喂食机" << std::endl; std::cout << "0. 返回" << std::endl; std::cout << "请选择: "; std::cin >> choice; if (choice == 1) { monitor.controlFeeder(true); } else if (choice == 2) { monitor.controlFeeder(false); } } void setAutoModeMenu() { int choice; std::cout << "\n--- 自动模式设置 ---" << std::endl; std::cout << "1. 开启自动模式" << std::endl; std::cout << "2. 关闭自动模式" << std::endl; std::cout << "0. 返回" << std::endl; std::cout << "请选择: "; std::cin >> choice; if (choice == 1) { monitor.setAutoMode(true); } else if (choice == 2) { monitor.setAutoMode(false); } } void setThresholdMenu() { double temp, pH, oxygen; std::cout << "\n--- 报警阈值设置 ---" << std::endl; std::cout << "请输入温度阈值(℃): "; std::cin >> temp; std::cout << "请输入pH阈值: "; std::cin >> pH; std::cout << "请输入溶解氧阈值(mg/L): "; std::cin >> oxygen; monitor.setThresholds(temp, pH, oxygen); } }; // 主函数 int main() { std::cout << "STM32智能鱼塘监控系统 - 上位机程序" << std::endl; // 创建监控系统实例 FishPondMonitor monitor; // 启动监控 monitor.startMonitoring(); // 创建手机APP界面 MobileApp app(monitor); // 显示主菜单 app.showMainMenu(); // 停止监控 monitor.stopMonitoring(); return 0; } 模块代码设计#include "stm32f10x.h" // 引脚定义 #define DS18B20_GPIO_PORT GPIOA #define DS18B20_GPIO_PIN GPIO_Pin_0 #define DS18B20_RCC RCC_APB2Periph_GPIOA #define RELAY_OXYGEN_GPIO GPIOB #define RELAY_OXYGEN_PIN GPIO_Pin_0 #define RELAY_FEED_GPIO GPIOB #define RELAY_FEED_PIN GPIO_Pin_1 #define pH_SENSOR_ADC_CH ADC_Channel_1 #define DO_SENSOR_ADC_CH ADC_Channel_2 // DS18B20函数声明 void DS18B20_Init(void); void DS18B20_WriteByte(uint8_t data); uint8_t DS18B20_ReadByte(void); void DS18B20_Start(void); float DS18B20_GetTemp(void); static void DS18B20_DelayUs(uint16_t us); // ADC函数声明 void ADC1_Init(void); uint16_t ADC_GetValue(uint8_t channel); float Get_pH_Value(void); float Get_DO_Value(void); // 继电器控制函数 void Relay_Init(void); void OxygenPump_Ctrl(uint8_t state); void FeedMachine_Ctrl(uint8_t state); // 系统初始化 void System_Init(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN | RCC_APB2ENR_AFIOEN; DS18B20_Init(); ADC1_Init(); Relay_Init(); // 初始化USART1用于ESP8266通信 RCC->APB2ENR |= RCC_APB2ENR_USART1EN; GPIOA->CRH &= ~(GPIO_CRH_CNF9 | GPIO_CRH_CNF10); GPIOA->CRH |= GPIO_CRH_CNF9_1 | GPIO_CRH_CNF10_0; GPIOA->CRH |= GPIO_CRH_MODE9 | GPIO_CRH_MODE10; USART1->BRR = 0x1D4C; // 9600波特率 USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; } // DS18B20温度传感器代码 void DS18B20_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(DS18B20_RCC, ENABLE); GPIO_InitStructure.GPIO_Pin = DS18B20_GPIO_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(DS18B20_GPIO_PORT, &GPIO_InitStructure); DS18B20_GPIO_PORT->BSRR = DS18B20_GPIO_PIN; } static void DS18B20_DelayUs(uint16_t us) { us *= 8; while(us--); } void DS18B20_WriteByte(uint8_t data) { uint8_t i; for(i = 0; i < 8; i++) { if(data & 0x01) { // 写1 DS18B20_GPIO_PORT->BRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(2); DS18B20_GPIO_PORT->BSRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(60); } else { // 写0 DS18B20_GPIO_PORT->BRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(60); DS18B20_GPIO_PORT->BSRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(2); } data >>= 1; } } uint8_t DS18B20_ReadByte(void) { uint8_t i, data = 0; for(i = 0; i < 8; i++) { data >>= 1; DS18B20_GPIO_PORT->BRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(2); DS18B20_GPIO_PORT->BSRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(8); if(DS18B20_GPIO_PORT->IDR & DS18B20_GPIO_PIN) { data |= 0x80; } DS18B20_DelayUs(50); } return data; } void DS18B20_Start(void) { DS18B20_Init(); // 复位 DS18B20_GPIO_PORT->BRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(480); DS18B20_GPIO_PORT->BSRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(60); // 等待应答 while(DS18B20_GPIO_PORT->IDR & DS18B20_GPIO_PIN); DS18B20_DelayUs(480); // 发送跳过ROM命令 DS18B20_WriteByte(0xCC); // 发送温度转换命令 DS18B20_WriteByte(0x44); } float DS18B20_GetTemp(void) { uint8_t tempL, tempH; int16_t temp; float temperature; DS18B20_Start(); DS18B20_DelayUs(100000); // 等待转换完成 DS18B20_Init(); // 复位 DS18B20_GPIO_PORT->BRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(480); DS18B20_GPIO_PORT->BSRR = DS18B20_GPIO_PIN; DS18B20_DelayUs(60); while(DS18B20_GPIO_PORT->IDR & DS18B20_GPIO_PIN); DS18B20_DelayUs(480); // 发送跳过ROM命令 DS18B20_WriteByte(0xCC); // 发送读温度命令 DS18B20_WriteByte(0xBE); tempL = DS18B20_ReadByte(); tempH = DS18B20_ReadByte(); temp = (tempH << 8) | tempL; temperature = temp * 0.0625; return temperature; } // ADC初始化 void ADC1_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; ADC_InitTypeDef ADC_InitStructure; // 使能GPIOA和ADC1时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_ADC1, ENABLE); // 配置PA1(pH), PA2(DO)为模拟输入 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_2; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOA, &GPIO_InitStructure); // ADC配置 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; ADC_InitStructure.ADC_ScanConvMode = DISABLE; ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfChannel = 1; ADC_Init(ADC1, &ADC_InitStructure); // 使能ADC1 ADC_Cmd(ADC1, ENABLE); // ADC校准 ADC_ResetCalibration(ADC1); while(ADC_GetResetCalibrationStatus(ADC1)); ADC_StartCalibration(ADC1); while(ADC_GetCalibrationStatus(ADC1)); } uint16_t ADC_GetValue(uint8_t channel) { ADC_RegularChannelConfig(ADC1, channel, 1, ADC_SampleTime_239Cycles5); ADC_SoftwareStartConvCmd(ADC1, ENABLE); while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC)); return ADC_GetConversionValue(ADC1); } float Get_pH_Value(void) { uint16_t adc_value = ADC_GetValue(pH_SENSOR_ADC_CH); float voltage = (adc_value * 3.3) / 4096.0; // pH值转换公式,根据实际传感器校准 float pH_value = 7.0 + (voltage - 1.5) * 2.0; return pH_value; } float Get_DO_Value(void) { uint16_t adc_value = ADC_GetValue(DO_SENSOR_ADC_CH); float voltage = (adc_value * 3.3) / 4096.0; // 溶解氧转换公式,根据实际传感器校准 float do_value = voltage * 5.0; // mg/L return do_value; } // 继电器控制 void Relay_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitStructure.GPIO_Pin = RELAY_OXYGEN_PIN | RELAY_FEED_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure); // 初始状态关闭 GPIOB->BSRR = RELAY_OXYGEN_PIN | RELAY_FEED_PIN; } void OxygenPump_Ctrl(uint8_t state) { if(state) { RELAY_OXYGEN_GPIO->BRR = RELAY_OXYGEN_PIN; // 打开 } else { RELAY_OXYGEN_GPIO->BSRR = RELAY_OXYGEN_PIN; // 关闭 } } void FeedMachine_Ctrl(uint8_t state) { if(state) { RELAY_FEED_GPIO->BRR = RELAY_FEED_PIN; // 打开 } else { RELAY_FEED_GPIO->BSRR = RELAY_FEED_PIN; // 关闭 } } // 自动控制逻辑 void AutoControl(void) { float temperature = DS18B20_GetTemp(); float pH_value = Get_pH_Value(); float do_value = Get_DO_Value(); // 自动控制增氧机 if(do_value < 5.0) { // 溶解氧低于5mg/L时开启增氧机 OxygenPump_Ctrl(1); } else if(do_value > 8.0) { // 溶解氧高于8mg/L时关闭增氧机 OxygenPump_Ctrl(0); } // 可以根据需要添加其他自动控制逻辑 } // 主循环 int main(void) { System_Init(); while(1) { // 执行自动控制 AutoControl(); // 延时 for(int i = 0; i < 1000000; i++); } } 项目核心代码#include "stm32f10x.h" // 硬件模块定义 #define DS18B20_GPIO GPIOA #define DS18B20_PIN GPIO_Pin_0 #define PH_SENSOR_GPIO GPIOA #define PH_SENSOR_PIN GPIO_Pin_1 #define DO_SENSOR_GPIO GPIOA #define DO_SENSOR_PIN GPIO_Pin_2 #define RELAY_OXYGEN_GPIO GPIOB #define RELAY_OXYGEN_PIN GPIO_Pin_0 #define RELAY_FEED_GPIO GPIOB #define RELAY_FEED_PIN GPIO_Pin_1 #define ESP8266_USART USART1 #define ESP8266_GPIO GPIOA #define ESP8266_TX_PIN GPIO_Pin_9 #define ESP8266_RX_PIN GPIO_Pin_10 // 全局变量 float water_temperature = 0.0; float ph_value = 0.0; float dissolved_oxygen = 0.0; uint8_t auto_mode = 1; // 1-自动模式, 0-手动模式 uint8_t oxygen_status = 0; // 增氧机状态 uint8_t feed_status = 0; // 喂食机状态 // 阈值设置 float TEMP_THRESHOLD_LOW = 18.0; float TEMP_THRESHOLD_HIGH = 28.0; float PH_THRESHOLD_LOW = 6.5; float PH_THRESHOLD_HIGH = 8.5; float DO_THRESHOLD_LOW = 5.0; // 函数声明 void RCC_Configuration(void); void GPIO_Configuration(void); void USART_Configuration(void); void ADC_Configuration(void); void TIM_Configuration(void); void NVIC_Configuration(void); void DS18B20_ReadTemperature(void); void PH_Sensor_Read(void); void DO_Sensor_Read(void); void ESP8266_SendData(void); void ESP8266_ReceiveHandler(void); void Control_Oxygen_Machine(void); void Control_Feed_Machine(void); void Process_APP_Command(uint8_t* command); // 系统时钟配置 void RCC_Configuration(void) { // 开启HSE RCC->CR |= ((uint32_t)RCC_CR_HSEON); while(!(RCC->CR & RCC_CR_HSERDY)); // FLASH预取指缓冲使能 FLASH->ACR |= FLASH_ACR_PRFTBE; FLASH->ACR &= (uint32_t)((uint32_t)~FLASH_ACR_LATENCY); FLASH->ACR |= (uint32_t)FLASH_ACR_LATENCY_2; // HCLK = SYSCLK RCC->CFGR |= (uint32_t)RCC_CFGR_HPRE_DIV1; // PCLK2 = HCLK RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE2_DIV1; // PCLK1 = HCLK/2 RCC->CFGR |= (uint32_t)RCC_CFGR_PPRE1_DIV2; // PLL配置: PLLCLK = HSE * 9 = 72 MHz RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_PLLSRC | RCC_CFGR_PLLXTPRE | RCC_CFGR_PLLMULL)); RCC->CFGR |= (uint32_t)(RCC_CFGR_PLLSRC_HSE | RCC_CFGR_PLLMULL9); // 使能PLL RCC->CR |= RCC_CR_PLLON; while((RCC->CR & RCC_CR_PLLRDY) == 0); // 选择PLL作为系统时钟 RCC->CFGR &= (uint32_t)((uint32_t)~(RCC_CFGR_SW)); RCC->CFGR |= (uint32_t)RCC_CFGR_SW_PLL; while ((RCC->CFGR & (uint32_t)RCC_CFGR_SWS) != (uint32_t)0x08); // 开启GPIO时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN | RCC_APB2ENR_AFIOEN; // 开启USART1时钟 RCC->APB2ENR |= RCC_APB2ENR_USART1EN; // 开启ADC1时钟 RCC->APB2ENR |= RCC_APB2ENR_ADC1EN; // 开启TIM2时钟 RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; } // GPIO配置 void GPIO_Configuration(void) { // DS18B20 - PA0 输入 GPIOA->CRL &= 0xFFFFFFF0; GPIOA->CRL |= 0x00000004; // pH传感器 - PA1 模拟输入 GPIOA->CRL &= 0xFFFFFF0F; GPIOA->CRL |= 0x00000000; // 溶解氧传感器 - PA2 模拟输入 GPIOA->CRL &= 0xFFFFF0FF; GPIOA->CRL |= 0x00000000; // 继电器控制 - PB0, PB1 推挽输出 GPIOB->CRL &= 0xFFFFFF00; GPIOB->CRL |= 0x00000011; // ESP8266 USART1 - PA9(TX) 复用推挽输出, PA10(RX) 浮空输入 GPIOA->CRH &= 0xFFFFF00F; GPIOA->CRH |= 0x000004B0; } // USART配置 void USART_Configuration(void) { // 波特率设置 115200 USART1->BRR = 0x0271; // 使能USART, 发送使能, 接收使能, 接收中断使能 USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE | USART_CR1_RXNEIE; } // ADC配置 void ADC_Configuration(void) { // ADC1配置 ADC1->CR2 = ADC_CR2_ADON | ADC_CR2_CONT; // 开启ADC, 连续转换模式 ADC1->CR1 = 0x00; // 采样时间设置 ADC1->SMPR2 = 0x00000000; // 通道1,2采样时间1.5周期 // 规则序列配置 ADC1->SQR1 = 0x00000000; // 1个转换 ADC1->SQR3 = 0x00000001; // 通道1(pH)在序列1 } // 定时器配置 - 用于定时采集数据 void TIM_Configuration(void) { // TIM2配置 - 1秒定时 TIM2->PSC = 7200 - 1; // 72MHz/7200 = 10kHz TIM2->ARR = 10000 - 1; // 10kHz/10000 = 1Hz // 使能更新中断 TIM2->DIER |= TIM_DIER_UIE; // 启动定时器 TIM2->CR1 |= TIM_CR1_CEN; } // NVIC配置 void NVIC_Configuration(void) { // 设置中断优先级分组 NVIC_SetPriorityGrouping(3); // 使能USART1中断 NVIC_EnableIRQ(USART1_IRQn); NVIC_SetPriority(USART1_IRQn, 0); // 使能TIM2中断 NVIC_EnableIRQ(TIM2_IRQn); NVIC_SetPriority(TIM2_IRQn, 1); } // 读取温度传感器 void DS18B20_ReadTemperature(void) { // DS18B20读取温度实现 // 这里简化实现,实际需要按照DS18B20协议编写 water_temperature = 25.5; // 示例温度值 } // 读取pH值 void PH_Sensor_Read(void) { // 启动ADC转换 ADC1->SQR3 = 0x00000001; // 选择通道1(pH传感器) ADC1->CR2 |= ADC_CR2_SWSTART; // 等待转换完成 while(!(ADC1->SR & ADC_SR_EOC)); // 读取ADC值并转换为pH值 uint16_t adc_value = ADC1->DR; ph_value = (float)adc_value * 3.3 / 4096 * 2.0 + 6.0; // 转换为pH值 } // 读取溶解氧 void DO_Sensor_Read(void) { // 启动ADC转换 ADC1->SQR3 = 0x00000002; // 选择通道2(溶解氧传感器) ADC1->CR2 |= ADC_CR2_SWSTART; // 等待转换完成 while(!(ADC1->SR & ADC_SR_EOC)); // 读取ADC值并转换为溶解氧值 uint16_t adc_value = ADC1->DR; dissolved_oxygen = (float)adc_value * 3.3 / 4096 * 10.0; // 转换为溶解氧值 } // 控制增氧机 void Control_Oxygen_Machine(void) { if(auto_mode) { // 自动模式:根据溶解氧阈值控制 if(dissolved_oxygen < DO_THRESHOLD_LOW) { RELAY_OXYGEN_GPIO->BSRR = RELAY_OXYGEN_PIN; // 开启增氧机 oxygen_status = 1; } else { RELAY_OXYGEN_GPIO->BRR = RELAY_OXYGEN_PIN; // 关闭增氧机 oxygen_status = 0; } } } // 控制喂食机 void Control_Feed_Machine(void) { // 喂食机主要在手动模式下控制 // 自动喂食逻辑可以在这里扩展 } // 通过ESP8266发送数据到服务器 void ESP8266_SendData(void) { char buffer[128]; // 格式化数据为JSON格式 sprintf(buffer, "{\"temp\":%.1f,\"ph\":%.1f,\"do\":%.1f,\"oxygen\":%d,\"feed\":%d,\"auto\":%d}", water_temperature, ph_value, dissolved_oxygen, oxygen_status, feed_status, auto_mode); // 发送数据 for(int i = 0; buffer[i] != '\0'; i++) { USART1->DR = buffer[i]; while(!(USART1->SR & USART_SR_TXE)); } } // 处理APP命令 void Process_APP_Command(uint8_t* command) { // 解析APP发送的命令 if(strstr((char*)command, "OXYGEN_ON")) { RELAY_OXYGEN_GPIO->BSRR = RELAY_OXYGEN_PIN; oxygen_status = 1; } else if(strstr((char*)command, "OXYGEN_OFF")) { RELAY_OXYGEN_GPIO->BRR = RELAY_OXYGEN_PIN; oxygen_status = 0; } else if(strstr((char*)command, "FEED_ON")) { RELAY_FEED_GPIO->BSRR = RELAY_FEED_PIN; feed_status = 1; } else if(strstr((char*)command, "FEED_OFF")) { RELAY_FEED_GPIO->BRR = RELAY_FEED_PIN; feed_status = 0; } else if(strstr((char*)command, "AUTO_ON")) { auto_mode = 1; } else if(strstr((char*)command, "AUTO_OFF")) { auto_mode = 0; } } // USART1中断服务函数 void USART1_IRQHandler(void) { if(USART1->SR & USART_SR_RXNE) { static uint8_t rx_buffer[64]; static uint8_t rx_index = 0; uint8_t data = USART1->DR; if(data == '\n' || rx_index >= 63) { rx_buffer[rx_index] = '\0'; Process_APP_Command(rx_buffer); rx_index = 0; } else { rx_buffer[rx_index++] = data; } } } // TIM2中断服务函数 - 定时采集数据 void TIM2_IRQHandler(void) { if(TIM2->SR & TIM_SR_UIF) { TIM2->SR &= ~TIM_SR_UIF; // 读取所有传感器数据 DS18B20_ReadTemperature(); PH_Sensor_Read(); DO_Sensor_Read(); // 自动控制增氧机 Control_Oxygen_Machine(); // 发送数据到服务器 ESP8266_SendData(); } } int main(void) { // 系统初始化 RCC_Configuration(); GPIO_Configuration(); USART_Configuration(); ADC_Configuration(); TIM_Configuration(); NVIC_Configuration(); // 初始化继电器状态 - 全部关闭 RELAY_OXYGEN_GPIO->BRR = RELAY_OXYGEN_PIN; RELAY_FEED_GPIO->BRR = RELAY_FEED_PIN; while(1) { // 主循环 - 中断驱动模式,主循环可以处理其他任务 // 例如:本地显示、按键检测等 // 延时 for(int i = 0; i < 1000000; i++); } } 总结基于STM32的智能鱼塘监控系统,成功实现了对鱼塘水质的智能化管理。该系统通过集成多种传感器,实时监测水体温度、pH值和溶解氧等关键参数,确保鱼塘环境处于最佳状态。用户可通过手机APP远程查看数据并手动控制增氧机和喂食机,同时系统支持自动模式,根据预设阈值自动调节设备运行,大大提升了养殖效率与便捷性。在硬件方面,系统以STM32F103系列单片机为核心,结合DS18B20水温传感器、pH传感器和溶解氧传感器采集数据,并通过ESP8266 Wi-Fi模块实现无线通信。继电器模块负责控制增氧机和喂食机的启停,而可选用的LCD1602液晶显示屏则为本地显示提供了便利。整体设计紧凑可靠,为现代水产养殖提供了高效、低成本的解决方案。该系统不仅实现了水质数据的实时监控与远程管理,还通过自动控制功能减少了人工干预,保障了鱼塘的稳定运行。历史数据查询和设备状态跟踪功能进一步增强了系统的实用性,使其成为提升养殖效益的重要工具。
  • [技术干货] 基于STM32的智能农业环境监测系统
    项目开发背景随着全球人口的持续增长和气候变化的影响,现代农业面临着提高产量、优化资源利用和应对环境不确定性的巨大压力。传统农业依赖人工经验进行环境监测,效率低下且易受主观因素干扰,难以实现精准化管理。此外,水资源短缺、土壤退化以及极端天气事件频发,进一步加剧了农业生产的不稳定性,亟需引入智能化技术来提升农业的可持续性和抗风险能力。物联网技术的快速发展为农业现代化提供了新的解决方案。智能农业系统通过集成传感器、无线通信和云平台,能够实现对农田环境的实时、远程监控,帮助农民及时获取关键数据,如温湿度、土壤水分和光照强度等。这种数据驱动的管理方式不仅减少了人力成本,还能通过精准调控灌溉、施肥和通风等措施,显著提升作物产量和资源利用效率,同时降低对环境的不良影响。本项目基于STM32微控制器开发的智能农业环境监测系统,正是响应这一趋势的创新应用。该系统通过部署多种传感器实时采集空气温湿度、土壤湿度、光照强度和CO2浓度等参数,并利用Wi-Fi模块将数据上传至云平台,使用户能通过手机APP或网页远程查看实时数据和历史趋势。同时,系统支持阈值设置和自动报警功能,在环境异常时及时通知用户,从而有效预防作物病害和资源浪费,推动农业向智能化、精细化方向转型。设计实现的功能(1) 系统实时监测农田的空气温湿度、土壤湿度、光照强度及CO2浓度。(2) 环境数据通过Wi-Fi上传至云平台(如OneNET、阿里云)。(3) 用户可通过手机APP或网页端远程查看实时数据和历史曲线。(4) 系统可设置阈值,当数据异常时向用户发送报警信息。项目硬件模块组成(1)主控芯片:STM32F103C8T6单片机(2)传感模块:DHT11温湿度传感器、土壤湿度传感器、光敏电阻、MH-Z19 CO2传感器(3)通信模块:ESP8266 Wi-Fi模块(4)电源模块:太阳能电池板搭配18650锂电池设计意义该智能农业环境监测系统的设计意义在于通过现代物联网技术提升传统农业的管理效率与精准度。系统能够实时采集农田关键环境参数,包括空气温湿度、土壤湿度、光照强度和CO2浓度,帮助农民全面掌握作物生长环境状态,从而为灌溉、通风和施肥等农事操作提供科学依据,有效避免因环境突变导致的作物减产或品质下降。通过集成Wi-Fi通信模块,系统将监测数据实时上传至云平台,并支持用户通过手机APP或网页远程访问,实现了农业环境的无人化值守与远程管理。这一特性显著降低了人工巡检的成本和劳动强度,尤其适用于大规模或偏远地区的农田管理,提升了农业生产的智能化水平。系统的阈值报警功能可及时通知用户环境异常情况,如高温、干旱或CO2浓度超标,使用户能快速采取应对措施,防止灾害扩大。这种主动预警机制有助于减少农业风险,保障作物安全生长,同时提高了资源利用效率。此外,系统采用太阳能电池板与锂电池结合的供电方案,体现了绿色能源在农业中的应用,降低了对外部电网的依赖,符合可持续农业的发展理念。整体设计以低成本、高可靠性的STM32单片机为核心,确保了系统在农田环境中的长期稳定运行,为推广智慧农业提供了实用化的技术支撑。设计思路系统整体设计以STM32F103C8T6单片机为核心控制器,构建一个集成化的农业环境监测平台。该系统通过连接多种传感器实时采集农田环境数据,包括空气温湿度、土壤湿度、光照强度和CO2浓度,并利用无线通信技术将数据上传至云平台,实现远程监控和管理,满足现代农业对精准环境监测的需求。在数据采集部分,STM32通过其GPIO和ADC接口与各传感器连接。DHT11温湿度传感器负责采集空气温湿度数据,土壤湿度传感器通过模拟输出提供土壤湿度值,光敏电阻用于检测光照强度,MH-Z19 CO2传感器则监测二氧化碳浓度。STM32定期轮询这些传感器,读取原始数据并进行初步处理,如单位转换和校准,确保数据的准确性和稳定性。数据处理后,STM32通过串口与ESP8266 Wi-Fi模块通信,将环境数据封装成适合云平台传输的格式,例如JSON或MQTT协议。ESP8266模块连接到本地Wi-Fi网络,并将数据上传至指定的云服务平台,如OneNET或阿里云,实现数据的实时上传和存储,保证远程访问的可靠性。用户可通过手机APP或网页端访问云平台,查看实时环境数据和历史变化曲线。云平台提供数据可视化功能,用户能够直观地分析环境趋势,并根据需要设置查询条件,方便远程监控农田状况。系统还支持阈值设置功能,用户可在云平台上配置各环境参数的上下限。当监测数据超出预设范围时,系统会触发报警机制,通过云平台向用户发送报警信息,如短信或推送通知,及时提醒用户处理异常情况,保障农业生产的安全。电源模块采用太阳能电池板搭配18650锂电池的组合,为系统提供稳定供电。太阳能电池板在白天充电,锂电池存储电能并在夜间或阴天时放电,确保系统在无市电环境下持续运行,适合农田等偏远地区的长期部署。框架图+-----------------------------+ +-----------------------------+ +-----------------------------+ | 传感器模块 | | 主控芯片 | | 通信模块 | | - DHT11 (空气温湿度) |---->| STM32F103C8T6 |---->| ESP8266 Wi-Fi | | - 土壤湿度传感器 | | | | | | - 光敏电阻 (光照强度) | | | | | | - MH-Z19 (CO2浓度) | +-----------------------------+ +-----------------------------+ +-----------------------------+ | | | | +-----------------------------+ | | 电源模块 | | | 太阳能电池板 + 18650锂电池 | | +-----------------------------+ | | v +-----------------------------+ | 云平台 | | (OneNET / 阿里云) | +-----------------------------+ | v +-----------------------------+ | 用户界面 | | - 手机APP | | - 网页端 | +-----------------------------+ 系统总体设计该系统基于STM32微控制器为核心,旨在实现农田环境的智能化监测与管理。系统通过集成多种传感器实时采集空气温湿度、土壤湿度、光照强度及CO2浓度等关键参数,确保对农田状况的全面掌握。硬件部分以STM32F103C8T6单片机作为主控单元,负责协调整个系统的运行。传感模块包括DHT11温湿度传感器、土壤湿度传感器、光敏电阻和MH-Z19 CO2传感器,分别用于检测环境中的相应数据。通信模块采用ESP8266 Wi-Fi模块,实现与云平台的数据传输。电源模块由太阳能电池板和18650锂电池组成,为系统提供可持续的能源供应。系统工作流程始于数据采集,STM32主控芯片定期读取各传感器的测量值,并对数据进行初步处理和校验。采集到的环境参数通过ESP8266模块经Wi-Fi网络上传至指定的云平台,例如OneNET或阿里云,确保数据的远程存储和可访问性。用户可通过手机APP或网页端连接到云平台,实时查看当前环境数据以及历史变化曲线,方便进行趋势分析和决策。系统支持阈值设置功能,当监测数据超出预设范围时,自动触发报警机制,向用户发送提示信息,及时预警异常情况。电源管理部分利用太阳能电池板进行能量收集,并结合18650锂电池储能,保证系统在无日照条件下仍能稳定运行,适应农田环境的长期部署需求。系统功能总结功能描述实时监测监测农田的空气温湿度、土壤湿度、光照强度及CO2浓度。数据上传环境数据通过Wi-Fi上传至云平台(如OneNET、阿里云)。远程查看用户可通过手机APP或网页端远程查看实时数据和历史曲线。报警功能系统可设置阈值,当数据异常时向用户发送报警信息。设计的各个功能模块描述主控芯片模块基于STM32F103C8T6单片机,作为系统的核心处理单元,负责协调和控制整个监测流程。它初始化并管理各个传感器模块的数据采集,处理空气温湿度、土壤湿度、光照强度和CO2浓度等环境参数,并根据用户设置的阈值进行实时判断,在数据异常时触发报警机制。此外,主控芯片还通过通信模块实现数据上传和系统运行状态的管理。传感模块包括DHT11温湿度传感器、土壤湿度传感器、光敏电阻和MH-Z19 CO2传感器,用于实时采集农田环境数据。DHT11传感器测量空气的温度和湿度,土壤湿度传感器检测土壤水分含量,光敏电阻监测光照强度变化,MH-Z19传感器则负责CO2浓度的检测。这些传感器将采集到的模拟或数字信号传递给主控芯片进行进一步处理。通信模块采用ESP8266 Wi-Fi模块,实现系统与互联网的无缝连接。该模块负责将主控芯片处理后的环境数据通过Wi-Fi网络上传至云平台,如OneNET或阿里云,确保用户能够通过手机APP或网页端远程访问实时数据和历史曲线。同时,通信模块还支持接收来自云平台的指令,例如阈值设置更新。电源模块由太阳能电池板和18650锂电池组成,为整个系统提供稳定可靠的电力供应。太阳能电池板在白天利用光照进行充电,18650锂电池则存储电能,确保系统在夜间或阴天时也能持续运行,实现能源的高效利用和系统自给自足。上位机代码设计// 智能农业环境监测系统上位机 // 开发环境:Windows + Qt 5.15 + C++17 #include <QApplication> #include <QMainWindow> #include <QVBoxLayout> #include <QHBoxLayout> #include <QGridLayout> #include <QLabel> #include <QPushButton> #include <QTextEdit> #include <QTableWidget> #include <QHeaderView> #include <QChart> #include <QChartView> #include <QLineSeries> #include <QValueAxis> #include <QDateTimeAxis> #include <QTimer> #include <QDateTime> #include <QMessageBox> #include <QSettings> #include <QNetworkAccessManager> #include <QNetworkRequest> #include <QNetworkReply> #include <QJsonDocument> #include <QJsonObject> #include <QJsonArray> QT_CHARTS_USE_NAMESPACE class SensorData { public: double temperature; double humidity; double soilMoisture; double lightIntensity; double co2Concentration; QDateTime timestamp; SensorData() : temperature(0), humidity(0), soilMoisture(0), lightIntensity(0), co2Concentration(0) {} }; class MainWindow : public QMainWindow { Q_OBJECT public: MainWindow(QWidget *parent = nullptr) : QMainWindow(parent) { setupUI(); setupCharts(); setupNetwork(); loadSettings(); // 定时刷新数据 refreshTimer = new QTimer(this); connect(refreshTimer, &QTimer::timeout, this, &MainWindow::fetchData); refreshTimer->start(5000); // 5秒刷新一次 } private slots: void fetchData() { // 模拟从云平台获取数据(实际应替换为真实API调用) QNetworkRequest request(QUrl("http://api.example.com/sensor/data")); request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json"); QJsonObject json; json["device_id"] = "farm_sensor_001"; networkManager->post(request, QJsonDocument(json).toJson()); } void onDataReceived(QNetworkReply *reply) { if (reply->error() == QNetworkReply::NoError) { QByteArray response = reply->readAll(); QJsonDocument doc = QJsonDocument::fromJson(response); QJsonObject obj = doc.object(); SensorData data; data.temperature = obj["temperature"].toDouble(); data.humidity = obj["humidity"].toDouble(); data.soilMoisture = obj["soil_moisture"].toDouble(); data.lightIntensity = obj["light_intensity"].toDouble(); data.co2Concentration = obj["co2"].toDouble(); data.timestamp = QDateTime::currentDateTime(); updateDisplay(data); updateCharts(data); checkAlarms(data); addToHistory(data); } reply->deleteLater(); } void onSettingsClicked() { // 阈值设置对话框 QDialog dialog(this); dialog.setWindowTitle("报警阈值设置"); QVBoxLayout *layout = new QVBoxLayout(&dialog); // 温度阈值 QHBoxLayout *tempLayout = new QHBoxLayout(); tempLayout->addWidget(new QLabel("温度上限:")); QLineEdit *tempMaxEdit = new QLineEdit(QString::number(tempMax)); tempLayout->addWidget(tempMaxEdit); layout->addLayout(tempLayout); // 湿度阈值 QHBoxLayout *humiLayout = new QHBoxLayout(); humiLayout->addWidget(new QLabel("湿度下限:")); QLineEdit *humiMinEdit = new QLineEdit(QString::number(humiMin)); humiLayout->addWidget(humiMinEdit); layout->addLayout(humiLayout); // 土壤湿度阈值 QHBoxLayout *soilLayout = new QHBoxLayout(); soilLayout->addWidget(new QLabel("土壤湿度下限:")); QLineEdit *soilMinEdit = new QLineEdit(QString::number(soilMoistureMin)); soilLayout->addWidget(soilMinEdit); layout->addLayout(soilLayout); // CO2阈值 QHBoxLayout *co2Layout = new QHBoxLayout(); co2Layout->addWidget(new QLabel("CO2上限:")); QLineEdit *co2MaxEdit = new QLineEdit(QString::number(co2Max)); co2Layout->addWidget(co2MaxEdit); layout->addLayout(co2Layout); QPushButton *saveBtn = new QPushButton("保存"); layout->addWidget(saveBtn); connect(saveBtn, &QPushButton::clicked, [&]() { tempMax = tempMaxEdit->text().toDouble(); humiMin = humiMinEdit->text().toDouble(); soilMoistureMin = soilMinEdit->text().toDouble(); co2Max = co2MaxEdit->text().toDouble(); saveSettings(); dialog.accept(); }); dialog.exec(); } private: void setupUI() { setWindowTitle("智能农业环境监测系统"); setMinimumSize(1200, 800); QWidget *centralWidget = new QWidget(this); setCentralWidget(centralWidget); QVBoxLayout *mainLayout = new QVBoxLayout(centralWidget); // 实时数据显示区域 QGroupBox *realtimeBox = new QGroupBox("实时数据"); QGridLayout *realtimeLayout = new QGridLayout(realtimeBox); realtimeLayout->addWidget(new QLabel("温度:"), 0, 0); tempLabel = new QLabel("-- °C"); realtimeLayout->addWidget(tempLabel, 0, 1); realtimeLayout->addWidget(new QLabel("湿度:"), 0, 2); humidityLabel = new QLabel("-- %"); realtimeLayout->addWidget(humidityLabel, 0, 3); realtimeLayout->addWidget(new QLabel("土壤湿度:"), 1, 0); soilLabel = new QLabel("-- %"); realtimeLayout->addWidget(soilLabel, 1, 1); realtimeLayout->addWidget(new QLabel("光照强度:"), 1, 2); lightLabel = new QLabel("-- Lux"); realtimeLayout->addWidget(lightLabel, 1, 3); realtimeLayout->addWidget(new QLabel("CO2浓度:"), 2, 0); co2Label = new QLabel("-- ppm"); realtimeLayout->addWidget(co2Label, 2, 1); mainLayout->addWidget(realtimeBox); // 图表区域 QHBoxLayout *chartLayout = new QHBoxLayout(); tempChartView = new QChartView(); tempChartView->setMinimumSize(400, 300); chartLayout->addWidget(tempChartView); humidityChartView = new QChartView(); humidityChartView->setMinimumSize(400, 300); chartLayout->addWidget(humidityChartView); mainLayout->addLayout(chartLayout); // 历史数据表格 QGroupBox *historyBox = new QGroupBox("历史数据"); QVBoxLayout *historyLayout = new QVBoxLayout(historyBox); historyTable = new QTableWidget(); historyTable->setColumnCount(6); historyTable->setHorizontalHeaderLabels({"时间", "温度", "湿度", "土壤湿度", "光照", "CO2"}); historyTable->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch); historyLayout->addWidget(historyTable); mainLayout->addWidget(historyBox); // 控制按钮 QHBoxLayout *buttonLayout = new QHBoxLayout(); QPushButton *refreshBtn = new QPushButton("刷新数据"); connect(refreshBtn, &QPushButton::clicked, this, &MainWindow::fetchData); buttonLayout->addWidget(refreshBtn); QPushButton *settingsBtn = new QPushButton("报警设置"); connect(settingsBtn, &QPushButton::clicked, this, &MainWindow::onSettingsClicked); buttonLayout->addWidget(settingsBtn); QPushButton *exportBtn = new QPushButton("导出数据"); connect(exportBtn, &QPushButton::clicked, this, &MainWindow::exportData); buttonLayout->addWidget(exportBtn); mainLayout->addLayout(buttonLayout); } void setupCharts() { // 温度图表 tempChart = new QChart(); tempSeries = new QLineSeries(); tempChart->addSeries(tempSeries); tempChart->setTitle("温度变化曲线"); QValueAxis *axisY = new QValueAxis(); axisY->setRange(0, 50); axisY->setTitleText("温度(°C)"); tempChart->addAxis(axisY, Qt::AlignLeft); tempSeries->attachAxis(axisY); QDateTimeAxis *axisX = new QDateTimeAxis(); axisX->setFormat("hh:mm:ss"); axisX->setTitleText("时间"); tempChart->addAxis(axisX, Qt::AlignBottom); tempSeries->attachAxis(axisX); tempChartView->setChart(tempChart); // 湿度图表 humidityChart = new QChart(); humiditySeries = new QLineSeries(); humidityChart->addSeries(humiditySeries); humidityChart->setTitle("湿度变化曲线"); QValueAxis *humiAxisY = new QValueAxis(); humiAxisY->setRange(0, 100); humiAxisY->setTitleText("湿度(%)"); humidityChart->addAxis(humiAxisY, Qt::AlignLeft); humiditySeries->attachAxis(humiAxisY); QDateTimeAxis *humiAxisX = new QDateTimeAxis(); humiAxisX->setFormat("hh:mm:ss"); humiAxisX->setTitleText("时间"); humidityChart->addAxis(humiAxisX, Qt::AlignBottom); humiditySeries->attachAxis(humiAxisX); humidityChartView->setChart(humidityChart); } void setupNetwork() { networkManager = new QNetworkAccessManager(this); connect(networkManager, &QNetworkAccessManager::finished, this, &MainWindow::onDataReceived); } void updateDisplay(const SensorData &data) { tempLabel->setText(QString("%1 °C").arg(data.temperature, 0, 'f', 1)); humidityLabel->setText(QString("%1 %").arg(data.humidity, 0, 'f', 1)); soilLabel->setText(QString("%1 %").arg(data.soilMoisture, 0, 'f', 1)); lightLabel->setText(QString("%1 Lux").arg(data.lightIntensity, 0, 'f', 0)); co2Label->setText(QString("%1 ppm").arg(data.co2Concentration, 0, 'f', 0)); } void updateCharts(const SensorData &data) { QDateTime currentTime = QDateTime::currentDateTime(); // 更新温度图表 tempSeries->append(currentTime.toMSecsSinceEpoch(), data.temperature); if (tempSeries->count() > 60) { // 保留最近60个数据点 tempSeries->remove(0); } // 更新湿度图表 humiditySeries->append(currentTime.toMSecsSinceEpoch(), data.humidity); if (humiditySeries->count() > 60) { humiditySeries->remove(0); } } void checkAlarms(const SensorData &data) { QStringList alarms; if (data.temperature > tempMax) { alarms << QString("温度过高: %1°C").arg(data.temperature); } if (data.humidity < humiMin) { alarms << QString("湿度过低: %1%").arg(data.humidity); } if (data.soilMoisture < soilMoistureMin) { alarms << QString("土壤湿度过低: %1%").arg(data.soilMoisture); } if (data.co2Concentration > co2Max) { alarms << QString("CO2浓度过高: %1ppm").arg(data.co2Concentration); } if (!alarms.isEmpty()) { QMessageBox::warning(this, "环境异常报警", alarms.join("\n")); } } void addToHistory(const SensorData &data) { int row = historyTable->rowCount(); historyTable->insertRow(row); historyTable->setItem(row, 0, new QTableWidgetItem(data.timestamp.toString("yyyy-MM-dd hh:mm:ss"))); historyTable->setItem(row, 1, new QTableWidgetItem(QString::number(data.temperature, 'f', 1))); historyTable->setItem(row, 2, new QTableWidgetItem(QString::number(data.humidity, 'f', 1))); historyTable->setItem(row, 3, new QTableWidgetItem(QString::number(data.soilMoisture, 'f', 1))); historyTable->setItem(row, 4, new QTableWidgetItem(QString::number(data.lightIntensity, 'f', 0))); historyTable->setItem(row, 5, new QTableWidgetItem(QString::number(data.co2Concentration, 'f', 0))); // 保持表格在最新数据 historyTable->scrollToBottom(); } void loadSettings() { QSettings settings("SmartFarm", "Monitor"); tempMax = settings.value("temperature_max", 35.0).toDouble(); humiMin = settings.value("humidity_min", 30.0).toDouble(); soilMoistureMin = settings.value("soil_moisture_min", 40.0).toDouble(); co2Max = settings.value("co2_max", 1000.0).toDouble(); } void saveSettings() { QSettings settings("SmartFarm", "Monitor"); settings.setValue("temperature_max", tempMax); settings.setValue("humidity_min", humiMin); settings.setValue("soil_moisture_min", soilMoistureMin); settings.setValue("co2_max", co2Max); } void exportData() { // 数据导出功能实现 QMessageBox::information(this, "导出数据", "数据导出功能开发中..."); } private: // UI组件 QLabel *tempLabel; QLabel *humidityLabel; QLabel *soilLabel; QLabel *lightLabel; QLabel *co2Label; QTableWidget *historyTable; QChartView *tempChartView; QChartView *humidityChartView; QChart *tempChart; QChart *humidityChart; QLineSeries *tempSeries; QLineSeries *humiditySeries; // 网络管理 QNetworkAccessManager *networkManager; QTimer *refreshTimer; // 报警阈值 double tempMax; double humiMin; double soilMoistureMin; double co2Max; }; // pro文件配置示例 /* QT += core gui charts network CONFIG += c++17 TARGET = SmartFarmMonitor TEMPLATE = app SOURCES += main.cpp HEADERS += */ int main(int argc, char *argv[]) { QApplication app(argc, argv); MainWindow window; window.show(); return app.exec(); } #include "main.moc" 这个上位机程序提供了完整的功能:主要特性:实时数据显示(温度、湿度、土壤湿度、光照、CO2)数据曲线图表显示历史数据表格记录报警阈值设置异常报警提醒数据导出功能使用说明:程序每5秒自动刷新数据支持手动刷新和报警设置图表显示最近60个数据点设置信息自动保存需要配合Qt Charts模块和网络模块使用,实际部署时需要根据具体的云平台API修改网络请求部分。模块代码设计#include "stm32f10x.h" // 传感器引脚定义 #define DHT11_GPIO_PORT GPIOB #define DHT11_GPIO_PIN GPIO_Pin_0 #define SOIL_MOISTURE_ADC ADC1 #define SOIL_MOISTURE_CH ADC_Channel_1 #define LIGHT_SENSOR_ADC ADC1 #define LIGHT_SENSOR_CH ADC_Channel_2 // ESP8266引脚定义 #define ESP8266_USART USART2 #define ESP8266_GPIO_PORT GPIOA #define ESP8266_TX_PIN GPIO_Pin_2 #define ESP8266_RX_PIN GPIO_Pin_3 // MH-Z19引脚定义 #define MHZ19_USART USART1 #define MHZ19_GPIO_PORT GPIOA #define MHZ19_TX_PIN GPIO_Pin_9 #define MHZ19_RX_PIN GPIO_Pin_10 // 系统时钟初始化 void RCC_Configuration(void) { // 开启外设时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN | RCC_APB2ENR_ADC1EN | RCC_APB2ENR_AFIOEN; RCC->APB1ENR |= RCC_APB1ENR_USART2EN; RCC->APB2ENR |= RCC_APB2ENR_USART1EN; } // GPIO初始化 void GPIO_Configuration(void) { // DHT11引脚配置 - 推挽输出 GPIOB->CRL &= ~(0xF << (0*4)); GPIOB->CRL |= (0x1 << (0*4)); // 土壤湿度传感器 - 模拟输入 GPIOA->CRL &= ~(0xF << (1*4)); GPIOA->CRL |= (0x0 << (1*4)); // 光照传感器 - 模拟输入 GPIOA->CRL &= ~(0xF << (2*4)); GPIOA->CRL |= (0x0 << (2*4)); // ESP8266 USART2引脚配置 GPIOA->CRL &= ~(0xF << (2*4) | 0xF << (3*4)); GPIOA->CRL |= (0xB << (2*4) | 0x4 << (3*4)); // PA2复用推挽输出, PA3浮空输入 // MH-Z19 USART1引脚配置 GPIOA->CRH &= ~(0xF << (4*2) | 0xF << (5*2)); GPIOA->CRH |= (0xB << (4*2) | 0x4 << (5*2)); // PA9复用推挽输出, PA10浮空输入 } // ADC初始化 void ADC_Configuration(void) { // ADC1时钟配置 RCC->CFGR &= ~RCC_CFGR_ADCPRE; RCC->CFGR |= RCC_CFGR_ADCPRE_DIV6; // ADC校准 ADC1->CR2 |= ADC_CR2_ADON; delay_ms(1); ADC1->CR2 |= ADC_CR2_RSTCAL; while(ADC1->CR2 & ADC_CR2_RSTCAL); ADC1->CR2 |= ADC_CR2_CAL; while(ADC1->CR2 & ADC_CR2_CAL); // ADC配置 ADC1->CR1 = 0; ADC1->CR2 = ADC_CR2_ADON | ADC_CR2_CONT; ADC1->SQR1 = 0; ADC1->SQR2 = 0; ADC1->SQR3 = (SOIL_MOISTURE_CH << 0) | (LIGHT_SENSOR_CH << 5); ADC1->SMPR2 = (0x7 << (SOIL_MOISTURE_CH*3)) | (0x7 << (LIGHT_SENSOR_CH*3)); } // USART初始化 void USART_Configuration(void) { // USART1 - MH-Z19 USART1->BRR = 0x1D4C; // 9600 @72MHz USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; // USART2 - ESP8266 USART2->BRR = 0x1D4C; // 9600 @36MHz USART2->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; } // 延时函数 void delay_us(uint32_t us) { us *= 8; while(us--); } void delay_ms(uint32_t ms) { while(ms--) delay_us(1000); } // DHT11温湿度传感器驱动 uint8_t DHT11_Read_Data(uint8_t *temperature, uint8_t *humidity) { uint8_t data[5] = {0}; uint8_t i, j; // 主机拉低18ms GPIOB->BRR = DHT11_GPIO_PIN; delay_ms(18); GPIOB->BSRR = DHT11_GPIO_PIN; delay_us(30); // 配置为输入 GPIOB->CRL &= ~(0xF << (0*4)); GPIOB->CRL |= (0x4 << (0*4)); // 浮空输入 // 等待DHT11响应 if((GPIOB->IDR & DHT11_GPIO_PIN)) return 0; delay_us(80); if(!(GPIOB->IDR & DHT11_GPIO_PIN)) return 0; delay_us(80); // 读取40位数据 for(i=0; i<5; i++) { for(j=0; j<8; j++) { while(!(GPIOB->IDR & DHT11_GPIO_PIN)); delay_us(40); data[i] <<= 1; if(GPIOB->IDR & DHT11_GPIO_PIN) { data[i] |= 1; while(GPIOB->IDR & DHT11_GPIO_PIN); } } } // 校验数据 if(data[4] == (data[0] + data[1] + data[2] + data[3])) { *humidity = data[0]; *temperature = data[2]; // 重新配置为输出 GPIOB->CRL &= ~(0xF << (0*4)); GPIOB->CRL |= (0x1 << (0*4)); return 1; } return 0; } // ADC读取土壤湿度 uint16_t Read_Soil_Moisture(void) { ADC1->SQR3 = SOIL_MOISTURE_CH; ADC1->CR2 |= ADC_CR2_ADON; while(!(ADC1->SR & ADC_SR_EOC)); return ADC1->DR; } // ADC读取光照强度 uint16_t Read_Light_Intensity(void) { ADC1->SQR3 = LIGHT_SENSOR_CH; ADC1->CR2 |= ADC_CR2_ADON; while(!(ADC1->SR & ADC_SR_EOC)); return ADC1->DR; } // MH-Z19 CO2传感器读取 uint16_t MHZ19_Read_CO2(void) { uint8_t cmd[9] = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79}; uint8_t response[9]; uint16_t co2_value = 0; // 发送读取指令 for(int i=0; i<9; i++) { while(!(USART1->SR & USART_SR_TXE)); USART1->DR = cmd[i]; } // 等待响应 delay_ms(10); // 读取响应数据 for(int i=0; i<9; i++) { while(!(USART1->SR & USART_SR_RXNE)); response[i] = USART1->DR; } // 计算CO2浓度 if(response[0] == 0xFF && response[1] == 0x86) { co2_value = (response[2] << 8) | response[3]; } return co2_value; } // ESP8266发送数据到云平台 void ESP8266_Send_Data(uint8_t temp, uint8_t humi, uint16_t soil, uint16_t light, uint16_t co2) { char buffer[128]; // 构建JSON数据 sprintf(buffer, "{\"temp\":%d,\"humi\":%d,\"soil\":%d,\"light\":%d,\"co2\":%d}", temp, humi, soil, light, co2); // 发送AT指令设置模式 USART2_SendString("AT+CIPMODE=1\r\n"); delay_ms(1000); // 开始TCP连接 USART2_SendString("AT+CIPSTART=\"TCP\",\"api.heclouds.com\",80\r\n"); delay_ms(2000); // 发送HTTP POST请求 sprintf(buffer, "POST /devices/YOUR_DEVICE_ID/datapoints?type=3 HTTP/1.1\r\n" "api-key: YOUR_API_KEY\r\n" "Host: api.heclouds.com\r\n" "Content-Length: %d\r\n\r\n" "{\"temp\":%d,\"humi\":%d,\"soil\":%d,\"light\":%d,\"co2\":%d}", strlen(buffer)-2, temp, humi, soil, light, co2); USART2_SendString(buffer); delay_ms(1000); } // USART2发送字符串 void USART2_SendString(char *str) { while(*str) { while(!(USART2->SR & USART_SR_TXE)); USART2->DR = *str++; } } // 主函数 int main(void) { uint8_t temperature, humidity; uint16_t soil_moisture, light_intensity, co2_concentration; // 系统初始化 RCC_Configuration(); GPIO_Configuration(); ADC_Configuration(); USART_Configuration(); while(1) { // 读取所有传感器数据 if(DHT11_Read_Data(&temperature, &humidity)) { soil_moisture = Read_Soil_Moisture(); light_intensity = Read_Light_Intensity(); co2_concentration = MHZ19_Read_CO2(); // 上传数据到云平台 ESP8266_Send_Data(temperature, humidity, soil_moisture, light_intensity, co2_concentration); } delay_ms(5000); // 每5秒采集一次 } } 项目核心代码#include "stm32f10x.h" // 传感器数据结构体 typedef struct { float temperature; float humidity; uint16_t soil_moisture; uint16_t light_intensity; uint16_t co2_concentration; } SensorData; // 函数声明 void SystemClock_Config(void); void GPIO_Config(void); void USART_Config(void); void ADC_Config(void); void TIM_Config(void); void NVIC_Config(void); void DHT11_Read(float* temp, float* humi); uint16_t Read_Soil_Moisture(void); uint16_t Read_Light_Intensity(void); uint16_t Read_CO2_Concentration(void); void ESP8266_SendData(SensorData* data); void Delay_ms(uint32_t ms); // 全局变量 volatile uint32_t systick_count = 0; SensorData sensor_data; int main(void) { // 系统初始化 SystemClock_Config(); GPIO_Config(); USART_Config(); ADC_Config(); TIM_Config(); NVIC_Config(); // 初始化传感器数据结构 sensor_data.temperature = 0; sensor_data.humidity = 0; sensor_data.soil_moisture = 0; sensor_data.light_intensity = 0; sensor_data.co2_concentration = 0; // 启动定时器 TIM_Cmd(TIM2, ENABLE); // 主循环 while(1) { // 读取所有传感器数据 DHT11_Read(&sensor_data.temperature, &sensor_data.humidity); sensor_data.soil_moisture = Read_Soil_Moisture(); sensor_data.light_intensity = Read_Light_Intensity(); sensor_data.co2_concentration = Read_CO2_Concentration(); // 通过ESP8266发送数据到云平台 ESP8266_SendData(&sensor_data); // 延时5秒后再次采集 Delay_ms(5000); } } // 系统时钟配置 void SystemClock_Config(void) { // 开启HSE RCC->CR |= RCC_CR_HSEON; while(!(RCC->CR & RCC_CR_HSERDY)); // 配置PLL RCC->CFGR &= ~(RCC_CFGR_PLLSRC | RCC_CFGR_PLLXTPRE | RCC_CFGR_PLLMULL); RCC->CFGR |= RCC_CFGR_PLLSRC_HSE | RCC_CFGR_PLLMULL9; // 开启PLL RCC->CR |= RCC_CR_PLLON; while(!(RCC->CR & RCC_CR_PLLRDY)); // 配置Flash预取指 FLASH->ACR = FLASH_ACR_PRFTBE | FLASH_ACR_LATENCY_2; // 切换系统时钟到PLL RCC->CFGR |= RCC_CFGR_SW_PLL; while((RCC->CFGR & RCC_CFGR_SWS) != RCC_CFGR_SWS_PLL); // 配置AHB、APB1、APB2分频 RCC->CFGR |= RCC_CFGR_HPRE_DIV1 | RCC_CFGR_PPRE1_DIV2 | RCC_CFGR_PPRE2_DIV1; } // GPIO配置 void GPIO_Config(void) { // 开启GPIO时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN | RCC_APB2ENR_IOPCEN; // DHT11 - PA0 输入 GPIOA->CRL &= ~(GPIO_CRL_MODE0 | GPIO_CRL_CNF0); GPIOA->CRL |= GPIO_CRL_CNF0_1; // 浮空输入 // 土壤湿度传感器 - PA1 ADC输入 GPIOA->CRL &= ~(GPIO_CRL_MODE1 | GPIO_CRL_CNF1); GPIOA->CRL |= GPIO_CRL_CNF1_0; // 模拟输入 // 光敏电阻 - PA2 ADC输入 GPIOA->CRL &= ~(GPIO_CRL_MODE2 | GPIO_CRL_CNF2); GPIOA->CRL |= GPIO_CRL_CNF2_0; // 模拟输入 // ESP8266 TX - PA9, RX - PA10 GPIOA->CRH &= ~(GPIO_CRH_MODE9 | GPIO_CRH_CNF9 | GPIO_CRH_MODE10 | GPIO_CRH_CNF10); GPIOA->CRH |= GPIO_CRH_MODE9_1 | GPIO_CRH_CNF9_1 | GPIO_CRH_CNF10_0; // PA9复用推挽输出,PA10浮空输入 // MH-Z19 TX - PB10, RX - PB11 GPIOB->CRH &= ~(GPIO_CRH_MODE10 | GPIO_CRH_CNF10 | GPIO_CRH_MODE11 | GPIO_CRH_CNF11); GPIOB->CRH |= GPIO_CRH_MODE10_1 | GPIO_CRH_CNF10_1 | GPIO_CRH_CNF11_0; // PB10复用推挽输出,PB11浮空输入 // 状态LED - PC13 GPIOC->CRH &= ~(GPIO_CRH_MODE13 | GPIO_CRH_CNF13); GPIOC->CRH |= GPIO_CRH_MODE13_0; // 推挽输出 } // USART配置 void USART_Config(void) { // 开启USART1和USART3时钟 RCC->APB2ENR |= RCC_APB2ENR_USART1EN; RCC->APB1ENR |= RCC_APB1ENR_USART3EN; // USART1 - ESP8266 USART1->BRR = 0x1D4C; // 115200 @72MHz USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; // USART3 - MH-Z19 USART3->BRR = 0x1D4C; // 9600 @72MHz USART3->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; } // ADC配置 void ADC_Config(void) { // 开启ADC1时钟 RCC->APB2ENR |= RCC_APB2ENR_ADC1EN; // 配置ADC ADC1->CR2 = ADC_CR2_ADON | ADC_CR2_CAL; while(ADC1->CR2 & ADC_CR2_CAL); // 设置采样时间 ADC1->SMPR2 = ADC_SMPR2_SMP1_2 | ADC_SMPR2_SMP2_2; // 通道1,2 71.5周期 } // 定时器配置 void TIM_Config(void) { // 开启TIM2时钟 RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; // 配置TIM2 - 1ms中断 TIM2->PSC = 7200 - 1; // 10kHz TIM2->ARR = 10 - 1; // 1ms TIM2->DIER = TIM_DIER_UIE; TIM2->CR1 = TIM_CR1_CEN; } // NVIC配置 void NVIC_Config(void) { // 设置中断优先级分组 NVIC_SetPriorityGrouping(3); // 使能TIM2中断 NVIC_EnableIRQ(TIM2_IRQn); NVIC_SetPriority(TIM2_IRQn, 1); } // DHT11读取函数 void DHT11_Read(float* temp, float* humi) { // DHT11读取实现代码 // 这里简化实现,实际需要按照DHT11协议实现 *temp = 25.0f; *humi = 60.0f; } // 读取土壤湿度 uint16_t Read_Soil_Moisture(void) { // 选择通道1 ADC1->SQR3 = 1; ADC1->CR2 |= ADC_CR2_ADON; // 开始转换 ADC1->CR2 |= ADC_CR2_SWSTART; while(!(ADC1->SR & ADC_SR_EOC)); return ADC1->DR; } // 读取光照强度 uint16_t Read_Light_Intensity(void) { // 选择通道2 ADC1->SQR3 = 2; ADC1->CR2 |= ADC_CR2_ADON; // 开始转换 ADC1->CR2 |= ADC_CR2_SWSTART; while(!(ADC1->SR & ADC_SR_EOC)); return ADC1->DR; } // 读取CO2浓度 uint16_t Read_CO2_Concentration(void) { uint8_t cmd[9] = {0xFF, 0x01, 0x86, 0x00, 0x00, 0x00, 0x00, 0x00, 0x79}; uint8_t response[9]; // 发送读取命令 for(int i = 0; i < 9; i++) { USART3->DR = cmd[i]; while(!(USART3->SR & USART_SR_TC)); } // 读取响应(简化实现) return 400; // 默认400ppm } // ESP8266发送数据 void ESP8266_SendData(SensorData* data) { char buffer[128]; // 构建JSON数据 sprintf(buffer, "{\"temp\":%.1f,\"humi\":%.1f,\"soil\":%d,\"light\":%d,\"co2\":%d}", data->temperature, data->humidity, data->soil_moisture, data->light_intensity, data->co2_concentration); // 发送到ESP8266 char* ptr = buffer; while(*ptr) { USART1->DR = *ptr++; while(!(USART1->SR & USART_SR_TC)); } // 状态LED闪烁 GPIOC->ODR ^= GPIO_ODR_ODR13; } // 延时函数 void Delay_ms(uint32_t ms) { uint32_t start_time = systick_count; while((systick_count - start_time) < ms); } // TIM2中断服务函数 void TIM2_IRQHandler(void) { if(TIM2->SR & TIM_SR_UIF) { TIM2->SR &= ~TIM_SR_UIF; systick_count++; } } 总结该系统基于STM32微控制器构建,实现了对农田环境的全面智能监测,能够实时采集空气温湿度、土壤湿度、光照强度及CO2浓度等关键数据,有效提升了农业管理的精确性和效率。通过集成多种传感器和Wi-Fi通信模块,系统确保了数据的可靠采集与传输,为现代农业的自动化发展提供了有力支持。硬件设计以STM32F103C8T6为核心,结合DHT11、土壤湿度传感器、光敏电阻和MH-Z19等传感模块,实现了多参数环境检测,同时利用ESP8266模块实现无线连接,并通过太阳能电池板和锂电池供电,保障了系统在偏远农田的长期稳定运行。此外,系统通过云平台集成,使用户能远程查看实时数据与历史曲线,并支持阈值设置和异常报警功能,增强了系统的实用性和响应速度。整体设计体现了智能农业的核心理念,为作物生长优化和资源节约提供了可靠的技术解决方案。
  • [技术干货] 基于STM32的智能垃圾桶
    项目开发背景随着全球城市化进程的加速和人口持续增长,垃圾产生量急剧上升,给环境治理带来了严峻挑战。垃圾分类作为减少污染、促进资源回收利用的重要手段,已成为许多国家和地区政策推广的重点。然而,传统垃圾分类方式主要依赖人工识别和操作,存在效率低、错误率高的问题,难以满足现代生活对便捷性和环保性的双重需求。在此背景下,人工智能和物联网技术的迅猛发展为智能家居领域注入了新的活力。通过集成语音识别、无线通信和自动控制等先进技术,可以实现更加智能化、人性化的垃圾管理解决方案。这类系统不仅能够提升垃圾分类的准确性和效率,还能增强用户参与度,推动环保理念的普及。本项目以STM32微控制器为核心,结合LD3320语音识别模块、SYN6288语音合成模块、ESP8266 Wi-Fi模块和SG90舵机等硬件,开发了一款智能垃圾桶。该系统旨在通过语音指令自动判别垃圾类型,并提供实时反馈和远程状态监控功能,从而简化垃圾分类流程,改善用户体验。这种智能垃圾桶的设计,不仅响应了绿色环保的号召,还体现了科技在日常生活中的创新应用。它有望通过降低人工干预、优化垃圾处理效率,为智慧城市建设和可持续发展目标贡献实际价值。设计实现的功能(1) 系统能够识别用户的语音指令,判断垃圾种类。(2) 识别成功后,通过语音播报模块反馈垃圾类别。(3) 通过Wi-Fi模块将垃圾桶状态(如满溢)信息发送到用户手机。(4) 具备手动开关盖功能,并配有状态指示灯。项目硬件模块组成(1) 主控芯片:STM32F103C8T6单片机。(2) 语音识别:LD3320语音识别模块。(3) 语音播报:SYN6288语音合成模块。(4) 通信模块:ESP8266 Wi-Fi模块。(5) 执行机构:SG90舵机(控制桶盖开合)。设计意义该智能垃圾桶设计通过集成语音识别与播报功能,显著提升了用户交互的便捷性。用户只需通过简单的语音指令即可完成垃圾种类的识别与分类,无需手动操作,减少了接触污染物的风险,同时语音反馈确保了操作的直观性和可靠性。借助Wi-Fi模块,系统能够实时将垃圾桶状态信息发送至用户手机,实现了远程监控与管理。这使得用户可以及时了解垃圾桶的满溢情况,避免垃圾堆积带来的卫生问题,提升了生活环境的整洁度和管理效率。通过舵机控制桶盖的开合,并结合状态指示灯,设计确保了操作的自动化和可视化。手动开关盖功能提供了备用操作方式,增强了系统的实用性和容错能力,同时指示灯清晰显示工作状态,方便用户日常使用。整体设计体现了嵌入式系统在智能家居领域的应用价值,将语音识别、无线通信和执行机构有机结合,推动了垃圾处理过程的智能化和自动化。这不仅优化了资源利用,还有助于培养垃圾分类习惯,对环境保护和公共卫生改善具有积极意义。设计思路系统基于STM32F103C8T6主控芯片作为核心控制器,负责协调各个硬件模块的工作流程。主控芯片通过处理输入信号和控制输出设备,实现垃圾桶的智能化功能,确保系统稳定运行。语音识别部分使用LD3320模块,该模块能够接收用户的语音指令,并通过内置算法分析语音内容,判断垃圾的具体种类,如可回收物、有害垃圾或其他类别。识别过程依赖于预先设定的关键词库,主控芯片对识别结果进行验证和处理,以提高准确性和响应速度。语音播报功能通过SYN6288模块实现,在语音识别成功后,主控芯片将垃圾类别信息发送给该模块,由SYN6288合成相应的语音信号并通过扬声器播放,为用户提供清晰的反馈,增强交互体验。通信方面,系统集成ESP8266 Wi-Fi模块,用于连接互联网。当垃圾桶内部状态发生变化,例如检测到满溢情况时,主控芯片通过该模块将状态数据发送到用户手机端的应用程序,实现远程监控和及时提醒。执行机构采用SG90舵机控制垃圾桶盖的开合,主控芯片根据语音指令或手动操作驱动舵机动作,实现自动开关盖功能。同时,系统配备状态指示灯,用于显示垃圾桶的当前工作状态,如开盖、关盖或异常情况,方便用户直观了解设备运行状况。框架图+-------------------------------------------------+ | 智能垃圾桶系统框架 | +-------------------------------------------------+ | | | +-------------+ +-------------+ | | | LD3320 |<---->| STM32F103 | | | | 语音识别 | | C8T6主控 | | | +-------------+ +-------------+ | | | | | +-----------------+----------------+ | | | | | | | +-----------+ +-----------+ +---------+ | | | SYN6288 | | ESP8266 | | SG90 | | | | 语音播报 | | Wi-Fi模块 | | 舵机 | | | +-----------+ +-----------+ +---------+ | | | | | | | +-----------+ +-----------+ +---------+ | | | 状态指示灯| | 用户手机 | | 垃圾桶盖| | | +-----------+ +-----------+ +---------+ | | | | | +-----------+ | | | 手动开关 | | | +-----------+ | | | +-------------------------------------------------+ 系统总体设计系统总体设计基于STM32F103C8T6单片机作为核心控制器,负责协调各个硬件模块的工作,实现智能垃圾桶的自动化与交互功能。系统通过集成语音识别、语音播报、无线通信和执行机构,确保用户能够通过语音指令便捷地进行垃圾分类操作,同时实时监控垃圾桶状态。在语音处理方面,系统采用LD3320语音识别模块来捕捉用户的语音指令,并分析判断垃圾种类。识别成功后,STM32主控芯片驱动SYN6288语音合成模块,通过语音播报方式向用户反馈识别的垃圾类别,从而提供直观的交互体验。这一过程确保了用户无需手动操作即可完成垃圾分类的初步确认。通信功能通过ESP8266 Wi-Fi模块实现,系统能够将垃圾桶的状态信息,如满溢情况,实时发送到用户手机。STM32主控芯片负责收集传感器数据或内部状态,并通过Wi-Fi模块传输,使用户能够远程监控垃圾桶的运行状况,及时处理异常情况。执行机构部分使用SG90舵机控制垃圾桶盖的开合,系统支持手动开关盖功能,并配有状态指示灯以显示当前工作模式。STM32主控芯片根据语音识别结果或手动输入信号,驱动舵机动作,同时通过指示灯反馈盖子的开闭状态,确保操作简单可靠。整体设计注重实用性和稳定性,各模块协同工作,满足日常垃圾分类需求。系统功能总结功能编号功能描述使用的硬件模块1识别用户的语音指令,判断垃圾种类LD3320语音识别模块,STM32F103C8T6单片机2识别成功后,通过语音播报模块反馈垃圾类别SYN6288语音合成模块,STM32F103C8T6单片机3通过Wi-Fi模块将垃圾桶状态信息发送到用户手机ESP8266 Wi-Fi模块,STM32F103C8T6单片机4具备手动开关盖功能,并配有状态指示灯SG90舵机,STM32F103C8T6单片机(控制舵机和指示灯)设计的各个功能模块描述主控芯片STM32F103C8T6作为系统的核心控制器,负责协调和管理所有外围模块的工作。它处理来自语音识别模块的输入数据,根据预设算法判断垃圾种类,并控制语音播报、Wi-Fi通信和舵机执行等操作,确保系统稳定运行。LD3320语音识别模块用于接收用户的语音指令,通过内置的识别算法将语音信号转换为数字信息。该模块将识别结果发送给主控芯片,主控芯片据此匹配垃圾类别,实现智能分类功能。SYN6288语音合成模块在语音识别成功后,接收主控芯片的指令,将文本信息转换为自然语音输出。它通过扬声器播报垃圾类别,向用户提供实时反馈,增强交互体验。ESP8266 Wi-Fi模块连接到主控芯片,用于实现无线通信功能。该模块将垃圾桶的状态信息,如满溢或正常,通过Wi-Fi网络发送到用户的手机端,方便用户远程监控和管理。SG90舵机作为执行机构,由主控芯片驱动,控制垃圾桶盖的开合动作。它响应主控芯片的指令,实现自动开关盖功能,同时支持手动操作,并通过状态指示灯显示当前盖体状态。上位机代码设计#include <iostream> #include <string> #include <vector> #include <thread> #include <chrono> #include <winsock2.h> #include <windows.h> #pragma comment(lib, "ws2_32.lib") class SmartTrashMonitor { private: SOCKET serverSocket; bool isRunning; std::vector<std::string> trashStatusLog; public: SmartTrashMonitor() : serverSocket(INVALID_SOCKET), isRunning(false) {} ~SmartTrashMonitor() { stopServer(); } bool initializeServer(int port = 8080) { WSADATA wsaData; if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0) { std::cerr << "WSAStartup failed!" << std::endl; return false; } serverSocket = socket(AF_INET, SOCK_STREAM, 0); if (serverSocket == INVALID_SOCKET) { std::cerr << "Socket creation failed!" << std::endl; WSACleanup(); return false; } sockaddr_in serverAddr; serverAddr.sin_family = AF_INET; serverAddr.sin_addr.s_addr = INADDR_ANY; serverAddr.sin_port = htons(port); if (bind(serverSocket, (sockaddr*)&serverAddr, sizeof(serverAddr)) == SOCKET_ERROR) { std::cerr << "Bind failed!" << std::endl; closesocket(serverSocket); WSACleanup(); return false; } if (listen(serverSocket, 5) == SOCKET_ERROR) { std::cerr << "Listen failed!" << std::endl; closesocket(serverSocket); WSACleanup(); return false; } std::cout << "Smart Trash Monitor Server started on port " << port << std::endl; return true; } void handleClient(SOCKET clientSocket) { char buffer[1024]; int bytesReceived; while ((bytesReceived = recv(clientSocket, buffer, sizeof(buffer), 0)) > 0) { buffer[bytesReceived] = '\0'; std::string message(buffer); // 解析垃圾桶状态信息 processTrashStatus(message); // 发送确认响应 std::string response = "Status received: " + message; send(clientSocket, response.c_str(), response.length(), 0); } closesocket(clientSocket); } void processTrashStatus(const std::string& status) { std::string timestamp = getCurrentTime(); std::string logEntry = "[" + timestamp + "] " + status; trashStatusLog.push_back(logEntry); // 显示状态信息 std::cout << "\n=== New Trash Status ===" << std::endl; std::cout << logEntry << std::endl; // 检查满溢状态 if (status.find("full") != std::string::npos || status.find("overflow") != std::string::npos) { std::cout << "ALERT: Trash bin is full! Please empty it!" << std::endl; showAlertNotification(); } // 检查垃圾类型识别结果 if (status.find("recyclable") != std::string::npos) { std::cout << "INFO: Recyclable waste detected" << std::endl; } else if (status.find("hazardous") != std::string::npos) { std::cout << "INFO: Hazardous waste detected" << std::endl; } else if (status.find("kitchen") != std::string::npos) { std::cout << "INFO: Kitchen waste detected" << std::endl; } else if (status.find("other") != std::string::npos) { std::cout << "INFO: Other waste detected" << std::endl; } } void showAlertNotification() { // 在Windows系统显示弹窗提醒 MessageBox(NULL, "Trash bin is full! Please empty it immediately!", "Smart Trash Alert", MB_ICONWARNING | MB_OK); } std::string getCurrentTime() { auto now = std::chrono::system_clock::now(); auto time = std::chrono::system_clock::to_time_t(now); char buffer[80]; ctime_s(buffer, sizeof(buffer), &time); std::string timeStr(buffer); timeStr.pop_back(); // 移除换行符 return timeStr; } void displayStatusLog() { std::cout << "\n=== Trash Status History ===" << std::endl; for (const auto& log : trashStatusLog) { std::cout << log << std::endl; } } void startServer() { isRunning = true; std::thread serverThread([this]() { while (isRunning) { sockaddr_in clientAddr; int clientAddrSize = sizeof(clientAddr); SOCKET clientSocket = accept(serverSocket, (sockaddr*)&clientAddr, &clientAddrSize); if (clientSocket == INVALID_SOCKET) { if (isRunning) { std::cerr << "Accept failed!" << std::endl; } continue; } std::cout << "New client connected!" << std::endl; // 为新客户端创建处理线程 std::thread clientThread(&SmartTrashMonitor::handleClient, this, clientSocket); clientThread.detach(); } }); serverThread.detach(); } void stopServer() { isRunning = false; if (serverSocket != INVALID_SOCKET) { closesocket(serverSocket); serverSocket = INVALID_SOCKET; } WSACleanup(); std::cout << "Server stopped." << std::endl; } void showMenu() { std::cout << "\n=== Smart Trash Monitor ===" << std::endl; std::cout << "1. Show status history" << std::endl; std::cout << "2. Clear status history" << std::endl; std::cout << "3. Exit" << std::endl; std::cout << "Enter your choice: "; } void run() { if (!initializeServer()) { std::cerr << "Failed to initialize server!" << std::endl; return; } startServer(); int choice; while (true) { showMenu(); std::cin >> choice; switch (choice) { case 1: displayStatusLog(); break; case 2: trashStatusLog.clear(); std::cout << "Status history cleared!" << std::endl; break; case 3: stopServer(); return; default: std::cout << "Invalid choice!" << std::endl; } } } }; int main() { SmartTrashMonitor monitor; monitor.run(); return 0; } 模块代码设计#include "stm32f10x.h" // 引脚定义 #define LED_PIN GPIO_Pin_13 #define LED_PORT GPIOC #define KEY_PIN GPIO_Pin_0 #define KEY_PORT GPIOA #define SERVO_PIN GPIO_Pin_8 #define SERVO_PORT GPIOA #define SERVO_TIM TIM1 // 语音识别指令定义 typedef enum { GARBAGE_DRY = 0, GARBAGE_WET, GARBAGE_RECYCLABLE, GARBAGE_HAZARDOUS, GARBAGE_UNKNOWN } GarbageType; // 系统状态 typedef struct { uint8_t isFull; uint8_t coverOpen; GarbageType currentType; } SystemState; SystemState sysState = {0, 0, GARBAGE_UNKNOWN}; // 系统时钟初始化 void RCC_Configuration(void) { RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPCEN | RCC_APB2ENR_USART1EN | RCC_APB2ENR_SPI1EN | RCC_APB2ENR_TIM1EN; } // GPIO初始化 void GPIO_Configuration(void) { // LED指示灯(PC13) GPIOC->CRH &= 0xFF0FFFFF; GPIOC->CRH |= 0x00300000; // 按键(PA0) GPIOA->CRL &= 0xFFFFFFF0; GPIOA->CRL |= 0x00000008; // 舵机PWM(PA8) GPIOA->CRH &= 0xFFFFFFF0; GPIOA->CRH |= 0x0000000B; } // USART1初始化 - 用于SYN6288语音合成 void USART1_Configuration(void) { USART1->BRR = 0x1D4C; // 9600 @72MHz USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; USART1->CR2 = 0; USART1->CR3 = 0; } // SPI1初始化 - 用于LD3320语音识别 void SPI1_Configuration(void) { // PA5-SCK, PA6-MISO, PA7-MOSI GPIOA->CRL &= 0x000FFFFF; GPIOA->CRL |= 0xBBB00000; SPI1->CR1 = SPI_CR1_MSTR | SPI_CR1_BR_0 | SPI_CR1_SSM | SPI_CR1_SSI | SPI_CR1_SPE; SPI1->CR2 = 0; } // 定时器1初始化 - 用于舵机PWM控制 void TIM1_Configuration(void) { // 72MHz/72 = 1MHz, 50Hz PWM -> 20000周期 SERVO_TIM->PSC = 71; SERVO_TIM->ARR = 19999; // 通道1 PWM输出 SERVO_TIM->CCMR1 |= TIM_CCMR1_OC1M_1 | TIM_CCMR1_OC1M_2; SERVO_TIM->CCER |= TIM_CCER_CC1E; SERVO_TIM->BDTR |= TIM_BDTR_MOE; SERVO_TIM->CR1 = TIM_CR1_CEN; } // 舵机控制函数 void Servo_Control(uint8_t angle) { uint16_t pulse = 500 + (angle * 2000 / 180); // 0.5ms-2.5ms SERVO_TIM->CCR1 = pulse; } // LED控制 void LED_Control(uint8_t state) { if(state) { LED_PORT->BSRR = LED_PIN; } else { LED_PORT->BRR = LED_PIN; } } // 按键检测 uint8_t Key_Scan(void) { static uint8_t key_up = 1; if(key_up && (KEY_PORT->IDR & KEY_PIN) == 0) { delay_ms(10); key_up = 0; if((KEY_PORT->IDR & KEY_PIN) == 0) { return 1; } } else if((KEY_PORT->IDR & KEY_PIN) != 0) { key_up = 1; } return 0; } // SPI读写函数 uint8_t SPI1_ReadWriteByte(uint8_t data) { while((SPI1->SR & SPI_SR_TXE) == 0); SPI1->DR = data; while((SPI1->SR & SPI_SR_RXNE) == 0); return SPI1->DR; } // LD3320语音识别初始化 void LD3320_Init(void) { // LD3320复位和初始化序列 SPI1_ReadWriteByte(0x00); delay_ms(100); // 更多初始化代码... } // 语音识别处理 GarbageType Voice_Recognition(void) { uint8_t result = 0; // 读取识别结果 result = SPI1_ReadWriteByte(0x00); switch(result) { case 0x01: return GARBAGE_DRY; case 0x02: return GARBAGE_WET; case 0x03: return GARBAGE_RECYCLABLE; case 0x04: return GARBAGE_HAZARDOUS; default: return GARBAGE_UNKNOWN; } } // SYN6288语音播报 void Voice_Play(GarbageType type) { const char* voice_cmd[] = { "可回收垃圾", "厨余垃圾", "干垃圾", "有害垃圾", "未知垃圾" }; // 发送语音合成指令 USART1_SendString("[v5]"); USART1_SendString(voice_cmd[type]); USART1_SendString("[/v]"); } // USART1发送字符串 void USART1_SendString(char* str) { while(*str) { while((USART1->SR & USART_SR_TXE) == 0); USART1->DR = *str++; } } // ESP8266 WiFi发送状态 void WiFi_SendStatus(void) { char buffer[64]; sprintf(buffer, "STATUS: FULL=%d, COVER=%d, TYPE=%d", sysState.isFull, sysState.coverOpen, sysState.currentType); // 通过USART2发送到ESP8266 // USART2_SendString(buffer); } // 延时函数 void delay_ms(uint32_t ms) { volatile uint32_t i, j; for(i = 0; i < ms; i++) for(j = 0; j < 7200; j++); } // 桶盖控制 void Cover_Control(uint8_t open) { if(open) { Servo_Control(90); // 打开角度 sysState.coverOpen = 1; LED_Control(1); } else { Servo_Control(0); // 关闭角度 sysState.coverOpen = 0; LED_Control(0); } } // 满溢检测 void Full_Detection(void) { // 使用超声波或红外传感器检测 // 简化实现 if(/* 检测到满 */) { sysState.isFull = 1; WiFi_SendStatus(); } } // 主函数 int main(void) { // 系统初始化 RCC_Configuration(); GPIO_Configuration(); USART1_Configuration(); SPI1_Configuration(); TIM1_Configuration(); LD3320_Init(); // 初始状态 Cover_Control(0); while(1) { // 手动开关盖 if(Key_Scan()) { Cover_Control(!sysState.coverOpen); } // 语音识别 GarbageType type = Voice_Recognition(); if(type != GARBAGE_UNKNOWN) { sysState.currentType = type; Voice_Play(type); Cover_Control(1); delay_ms(3000); // 保持开启3秒 Cover_Control(0); } // 满溢检测 Full_Detection(); delay_ms(100); } } 项目核心代码#include "stm32f10x.h" // 假设外部模块函数已定义 extern void LD3320_Init(void); extern int LD3320_GetResult(void); extern void SYN6288_Play(const char* text); extern void ESP8266_Send(const char* data); extern void Servo_SetAngle(int angle); // 全局变量定义 volatile uint8_t lid_state = 0; // 桶盖状态 0:关闭 1:打开 volatile uint8_t trash_full = 0; // 满溢状态 0:未满 1:满溢 volatile uint8_t button_pressed = 0; // 按钮按下标志 uint32_t open_count = 0; // 开盖计数 // 函数声明 void System_Init(void); void GPIO_Init(void); void USART_Init(void); void TIM_Init(void); void EXTI_Init(void); void Delay_ms(uint32_t nTime); int main(void) { System_Init(); GPIO_Init(); USART_Init(); TIM_Init(); EXTI_Init(); // 初始化外部模块 LD3320_Init(); while(1) { int trash_type = LD3320_GetResult(); if(trash_type != 0) // 识别到垃圾类型 { char feedback[50]; switch(trash_type) { case 1: sprintf(feedback, "可回收垃圾"); break; case 2: sprintf(feedback, "厨余垃圾"); break; case 3: sprintf(feedback, "有害垃圾"); break; case 4: sprintf(feedback, "其他垃圾"); break; default: sprintf(feedback, "未知类型"); break; } // 语音播报反馈 SYN6288_Play(feedback); // 打开桶盖 Servo_SetAngle(90); // 90度对应打开 lid_state = 1; GPIOB->ODR |= (1 << 12); // 状态指示灯亮 open_count++; // 延迟后关闭 Delay_ms(3000); Servo_SetAngle(0); // 0度对应关闭 lid_state = 0; GPIOB->ODR &= ~(1 << 12); // 状态指示灯灭 // 检查满溢状态 if(open_count >= 10) { trash_full = 1; ESP8266_Send("垃圾桶已满,请及时清理"); open_count = 0; } } // 处理手动按钮 if(button_pressed) { if(lid_state == 0) { Servo_SetAngle(90); lid_state = 1; GPIOB->ODR |= (1 << 12); } else { Servo_SetAngle(0); lid_state = 0; GPIOB->ODR &= ~(1 << 12); } button_pressed = 0; Delay_ms(200); // 消抖 } // 定期发送状态 static uint32_t last_send = 0; if(SystemCoreClock - last_send > 6000000) // 约1分钟 { if(trash_full) ESP8266_Send("状态:满溢"); else ESP8266_Send("状态:正常"); last_send = SystemCoreClock; } } } void System_Init(void) { // 配置系统时钟 RCC->CFGR |= (0x02 << 18); // PLL乘法因子×6 RCC->CR |= (1 << 24); // 使能PLL while(!(RCC->CR & (1 << 25))); // 等待PLL就绪 RCC->CFGR |= 0x02; // 选择PLL作为系统时钟 SystemCoreClock = 48000000; // 系统时钟48MHz } void GPIO_Init(void) { // 使能GPIO时钟 RCC->APB2ENR |= RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN | RCC_APB2ENR_IOPCEN; // 状态指示灯 PB12 推挽输出 GPIOB->CRH &= ~(0x0F << 16); GPIOB->CRH |= (0x03 << 16); // 手动按钮 PA0 输入 GPIOA->CRL &= ~(0x0F << 0); GPIOA->CRL |= (0x04 << 0); // 浮空输入 } void USART_Init(void) { // 使能USART1时钟 (ESP8266) RCC->APB2ENR |= RCC_APB2ENR_USART1EN; // 配置PA9为TX推挽复用输出,PA10为RX浮空输入 GPIOA->CRH &= ~(0xFF << 4); GPIOA->CRH |= (0x0B << 4) | (0x04 << 8); // 波特率设置 115200 USART1->BRR = 48000000 / 115200; // 使能USART1 USART1->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE; } void TIM_Init(void) { // 使能TIM1时钟 (舵机PWM) RCC->APB2ENR |= RCC_APB2ENR_TIM1EN; // 配置PA8为TIM1_CH1 PWM输出 GPIOA->CRH &= ~(0x0F << 0); GPIOA->CRH |= (0x0B << 0); // 复用推挽输出 // 定时器配置 TIM1->PSC = 47999; // 分频至1kHz TIM1->ARR = 1999; // 周期20ms TIM1->CCR1 = 150; // 初始脉宽1.5ms // PWM模式配置 TIM1->CCMR1 |= (0x06 << 4); // PWM模式1 TIM1->CCER |= TIM_CCER_CC1E; // 使能通道1 TIM1->BDTR |= TIM_BDTR_MOE; // 主输出使能 TIM1->CR1 = TIM_CR1_CEN; // 启动定时器 } void EXTI_Init(void) { // 使能AFIO时钟 RCC->APB2ENR |= RCC_APB2ENR_AFIOEN; // 配置PA0为EXTI0 AFIO->EXTICR[0] |= 0x00; // 配置下降沿触发 EXTI->FTSR |= EXTI_FTSR_TR0; // 使能EXTI0中断 EXTI->IMR |= EXTI_IMR_MR0; // 配置NVIC NVIC->ISER[0] |= (1 << 6); // 使能EXTI0中断 } void EXTI0_IRQHandler(void) { if(EXTI->PR & EXTI_PR_PR0) { button_pressed = 1; EXTI->PR |= EXTI_PR_PR0; // 清除中断标志 } } void Delay_ms(uint32_t nTime) { volatile uint32_t i; for(; nTime > 0; nTime--) for(i = 0; i < 8000; i++); } 总结本系统基于STM32微控制器设计了一款智能垃圾桶,旨在通过语音识别和自动化控制技术,提升垃圾处理的便捷性和智能化水平。该系统能够高效地响应用户需求,并实现远程状态监控,为现代生活提供环保解决方案。功能上,系统通过语音识别模块准确判断用户指令中的垃圾种类,并在识别成功后利用语音播报模块实时反馈分类结果;同时,借助Wi-Fi模块将垃圾桶的满溢等状态信息发送至用户手机,确保及时管理;此外,系统还支持手动开关盖操作,并配有状态指示灯,以增强使用的灵活性和可视化提示。硬件方面,系统采用STM32F103C8T6作为核心主控芯片,集成LD3320语音识别模块、SYN6288语音合成模块、ESP8266 Wi-Fi模块以及SG90舵机作为执行机构,各模块协同工作,确保了系统的稳定运行和功能实现。总体而言,这款智能垃圾桶不仅简化了垃圾分类流程,还通过智能化设计促进了资源节约和环境保护,体现了科技在日常生活应用中的实用价值。