【STM32实战】三模联动智能药盒:从传感器融合到云平台交互
1. 三模联动智能药盒的设计初衷家里老人经常忘记吃药或者药品存放不当导致变质这种场景可能很多人都遇到过。传统的药盒功能单一无法满足现代家庭对药品管理的需求。这正是我们设计这款三模联动智能药盒的初衷——用STM32为核心打造一个集环境监测、智能提醒和远程管理于一体的解决方案。在实际开发过程中我发现最大的挑战是如何让多种传感器协同工作。比如温湿度传感器DHT11的响应速度较慢而称重模块HX711需要稳定的供电环境红外传感器又容易受到环境光干扰。经过多次调试最终确定了以STM32F103C8T6为主控的方案这款芯片性价比高外设丰富完全能满足我们的需求。2. 硬件架构详解2.1 核心控制器选型STM32F103C8T6是我们最终选择的主控芯片江湖人称蓝莓派。它拥有72MHz的主频、64KB Flash和20KB RAM完全够用。更重要的是它支持多路ADC和PWM输出这对我们的传感器和舵机控制非常关键。我特别看重的是它的低功耗特性在待机模式下电流可以降到2μA左右。这意味着即使用电池供电也能维持很长时间的工作。实际测试中配合合理的电源管理系统可以连续工作30天以上。2.2 传感器阵列配置传感器是系统的感官我们选用了以下几种DHT11温湿度传感器监测药品存储环境HX711称重模块检测药品余量红外传感器检测用户取药动作光敏电阻环境光照检测这里有个小技巧HX711称重模块需要特别注意安装方式。我最初直接固定在PCB上结果发现读数波动很大。后来改用弹性支撑结构读数稳定性立即提升了80%。2.3 执行机构设计执行部分主要包括SG90舵机控制药盒开关有源蜂鸣器报警提示LED照明夜间取药辅助舵机的控制需要特别注意。我最初使用简单的延时控制发现动作不流畅。后来改用PWM渐变控制让药盒开关动作更加自然。具体实现是这样的void Servo_SmoothMove(uint8_t from, uint8_t to, uint16_t duration) { int16_t step (to from) ? 1 : -1; for(uint8_t posfrom; pos!to; posstep){ Servo_SetAngle(pos); delay_ms(duration/abs(to-from)); } }3. 三模控制系统的实现3.1 远程控制模式远程模式通过ESP8266-01S WiFi模块实现。我们选择了机智云平台它提供了完整的IoT解决方案。在实际部署时遇到最大的坑是WiFi模块的供电问题——ESP8266在发射时瞬时电流可能达到200mA如果电源设计不当会导致系统重启。解决方案是使用独立的LDO给WiFi模块供电在电源端增加470μF的电解电容软件上采用分时策略避免传感器采集和WiFi通信同时进行数据传输采用JSON格式一个典型的数据包如下{ temp: 25, humi: 60, weight: 120, box_open: false, alert: false }3.2 自动控制模式自动模式的核心是阈值判断。我们设计了一个灵活的状态机来处理各种情况typedef enum { STATE_NORMAL, STATE_TEMP_ALERT, STATE_HUMI_ALERT, STATE_WEIGHT_ALERT, STATE_MULTI_ALERT } AlertState; void Alert_Handler(void) { static AlertState state STATE_NORMAL; // 状态转移判断 if(bufe[3]TempYu bufe[4]HumiYu bufe[0]ZhongYu){ state STATE_MULTI_ALERT; } else if(bufe[3]TempYu){ state STATE_TEMP_ALERT; } // 其他条件判断... // 执行对应动作 switch(state){ case STATE_TEMP_ALERT: Buzzer_Beep(3, 200); break; // 其他状态处理... } }3.3 手动控制模式手动模式看似简单但要做好用户体验并不容易。我们设计了双重按键检测机制短按切换LED灯状态长按3秒切换工作模式按键检测代码采用了状态机设计有效消除了抖动问题typedef enum { BTN_IDLE, BTN_PRESSED, BTN_RELEASED, BTN_LONG_PRESS } BtnState; BtnState Key_Detect(uint8_t pin) { static uint32_t pressTime 0; static BtnState state BTN_IDLE; if(KEY_READ(pin) 0){ // 按键按下 if(state BTN_IDLE){ pressTime HAL_GetTick(); state BTN_PRESSED; } else if(state BTN_PRESSED (HAL_GetTick()-pressTime)3000){ state BTN_LONG_PRESS; return state; } } else{ // 按键释放 if(state BTN_PRESSED){ state BTN_RELEASED; return state; } state BTN_IDLE; } return BTN_IDLE; }4. 传感器数据融合算法4.1 数据预处理原始传感器数据往往存在噪声我们采用了多种滤波算法对于温湿度数据使用滑动平均滤波#define FILTER_LEN 5 uint8_t temp_filter_buf[FILTER_LEN]; uint8_t Temp_Filter(uint8_t new_val) { static uint8_t index 0; uint16_t sum 0; temp_filter_buf[index] new_val; if(index FILTER_LEN) index 0; for(uint8_t i0; iFILTER_LEN; i){ sum temp_filter_buf[i]; } return sum/FILTER_LEN; }对于称重数据采用中值滤波与卡尔曼滤波结合的方式有效消除了瞬时干扰。4.2 多传感器数据融合我们开发了一个简单的传感器置信度评估机制为每个传感器分配一个置信度系数0-1当多个传感器数据冲突时采用置信度加权平均动态调整置信度如连续5次异常读数则降低置信度具体实现typedef struct { float value; float confidence; uint8_t error_count; } SensorData; void Sensor_Fusion(SensorData* temp, SensorData* humi, SensorData* weight) { // 异常检测 if(temp-value 50 || temp-value 0){ temp-error_count; if(temp-error_count 5){ temp-confidence * 0.9; } } else { temp-error_count 0; } // 类似处理其他传感器... // 加权融合 float total_conf temp-confidence humi-confidence weight-confidence; bufe[3] (temp-value*temp-confidence humi-value*humi-confidence) / (temp-confidence humi-confidence); bufe[0] weight-value; }5. 低功耗设计与优化5.1 电源管理策略为了延长电池寿命我们设计了多级电源管理主控制器在无操作时进入Stop模式传感器采用分时供电WiFi模块仅在需要通信时唤醒关键实现代码void Enter_LowPower(void) { // 关闭外设时钟 __HAL_RCC_GPIOA_CLK_DISABLE(); // 保留必要外设 // 配置唤醒源 HAL_PWR_EnableWakeUpPin(PWR_WAKEUP_PIN1); // 进入Stop模式 HAL_PWR_EnterSTOPMode(PWR_LOWPOWERREGULATOR_ON, PWR_STOPENTRY_WFI); // 唤醒后重新初始化 SystemClock_Config(); MX_GPIO_Init(); }5.2 任务调度优化我们采用了时间片轮询的轻量级调度方案typedef struct { void (*task)(void); uint32_t interval; uint32_t last_run; } Task; Task task_list[] { {Sensor_Update, 1000, 0}, {Alert_Check, 2000, 0}, {Comm_Process, 5000, 0} }; void Scheduler_Run(void) { uint32_t now HAL_GetTick(); for(uint8_t i0; isizeof(task_list)/sizeof(Task); i){ if(now - task_list[i].last_run task_list[i].interval){ task_list[i].task(); task_list[i].last_run now; } } }6. 云平台交互实现6.1 数据传输协议我们定义了精简的通信协议上行数据设备→云传感器数据 状态信息下行数据云→设备控制命令 配置参数协议头定义#pragma pack(1) typedef struct { uint8_t start_flag; // 0xAA uint8_t cmd_type; uint16_t data_len; uint8_t checksum; } ProtocolHeader; #pragma pack()6.2 断网处理机制考虑到网络不稳定的情况我们实现了本地缓存机制网络正常时实时上传数据网络异常时数据存入Flash网络恢复后批量上传缓存数据关键实现#define MAX_CACHE 50 typedef struct { uint32_t timestamp; float temp; float humi; uint16_t weight; } SensorRecord; void Data_Cache(SensorRecord record) { static SensorRecord cache[MAX_CACHE]; static uint8_t count 0; if(WiFi_IsConnected()){ Cloud_Upload(record, 1); } else { if(count MAX_CACHE){ cache[count] record; } else { Flash_Write(cache, sizeof(cache)); count 0; } } }7. 产品化思考与改进方向经过实际使用测试我发现还有几个可以优化的地方首先是称重模块的校准问题。目前的校准过程还不够友好需要用户手动放置标准重量。下一步计划加入自动校准功能通过记录每次取药后的重量变化来自动更新参数。其次是药盒的机械结构。现版本的舵机直接驱动上盖长期使用后会出现磨损。新的设计考虑采用滑盖结构配合霍尔传感器检测开关状态。最后是用户界面。OLED显示屏在强光下可视性较差考虑增加自动亮度调节功能或者改用电子墨水屏。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2435950.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!