从跌倒检测到平衡小车:用ADXL345传感器玩转STM32的几种实战应用

张开发
2026/4/16 4:07:13 15 分钟阅读

分享文章

从跌倒检测到平衡小车:用ADXL345传感器玩转STM32的几种实战应用
从跌倒检测到平衡小车用ADXL345传感器玩转STM32的几种实战应用当你已经成功驱动ADXL345传感器并能够读取XYZ三轴数据时是否曾思考过这些数字背后隐藏的无限可能在创客的世界里传感器从来不只是冰冷的数据采集器而是连接物理世界与数字世界的魔法棒。本文将带你跳出基础驱动的舒适区探索ADXL345在STM32平台上的三个惊艳应用从保护老人安全的跌倒检测装置到酷炫的自平衡小车再到打破传统输入方式的体感游戏控制器。每个项目都配有核心算法解析和可落地的代码片段让你手中的传感器真正活起来。1. 跌倒检测报警器当科技守护生命跌倒对老年人造成的威胁远超想象。根据世界卫生组织数据全球每年有超过37万人因跌倒致死。利用ADXL345制作的原型系统可以在成本不到百元的情况下实现专业级跌倒监测功能的核心算法。1.1 原理与算法设计跌倒检测的核心是识别突然的加速度变化与后续的姿态异常。ADXL345的±16g量程正好满足捕捉剧烈运动的需求。我们需要重点关注三个关键参数合加速度矢量√(x²y²z²)姿态角度atan2(y, √(x²z²))冲击持续时间高加速度的持续采样点数// 合加速度计算示例 float calculate_total_accel(int16_t x, int16_t y, int16_t z) { float g_x (float)x * 0.0039; // 转换为重力加速度单位(±16g量程) float g_y (float)y * 0.0039; float g_z (float)z * 0.0039; return sqrt(g_x*g_x g_y*g_y g_z*g_z); }1.2 状态机实现一个可靠的检测系统应该包含以下状态状态触发条件响应动作正常合加速度2g持续监测冲击合加速度≥3g持续50ms启动倒计时跌倒冲击后5秒内姿态角60°触发警报误报冲击后恢复直立姿态返回正常// 简易状态机实现 typedef enum {NORMAL, IMPACT, FALL, FALSE_ALARM} State; State current_state NORMAL; uint32_t impact_time 0; void detect_fall(int16_t x, int16_t y, int16_t z) { float accel calculate_total_accel(x,y,z); float angle atan2(y, sqrt(x*xz*z)) * 180/M_PI; switch(current_state) { case NORMAL: if(accel 3.0) { current_state IMPACT; impact_time HAL_GetTick(); } break; case IMPACT: if(accel 1.0 fabs(angle) 15) { current_state FALSE_ALARM; } else if(HAL_GetTick() - impact_time 5000) { if(fabs(angle) 60) { current_state FALL; trigger_alarm(); } else { current_state NORMAL; } } break; // 其他状态处理... } }1.3 系统集成与优化完整的系统还需要考虑低功耗设计利用ADXL345的运动唤醒功能无线报警通过HC-05蓝牙模块发送警报到手机误报过滤加入移动平均滤波算法提示实际部署时需要针对不同体型的人进行阈值校准可以考虑增加学习模式让用户录入正常活动数据。2. 平衡小车从传感器到PID控制将ADXL345用于平衡小车是理解惯性测量单元(IMU)最直观的方式。相比完整的6轴IMU单用加速度计虽然存在局限但正是这种限制能帮助我们深入理解姿态估计的本质。2.1 姿态解算基础仅用加速度计计算倾角有其物理限制静态时倾角 atan2(accY, accZ)动态时加速度干扰会导致虚假重力现象互补滤波是解决这一问题的经典方案angle 0.98*(angle gyro*dt) 0.02*acc_angle虽然我们没有陀螺仪但可以通过以下方式优化#define ALPHA 0.1 // 滤波系数 float estimated_angle 0; void update_angle(int16_t y, int16_t z) { float acc_angle atan2(y, z) * 180/M_PI; static float last_angle 0; float gyro_estimate (acc_angle - last_angle) / 0.01; // 假设采样率100Hz last_angle acc_angle; estimated_angle (1-ALPHA)*(estimated_angle gyro_estimate*0.01) ALPHA*acc_angle; }2.2 电机控制实现平衡小车的核心是PID控制器主要参数包括比例项(P)与倾角成正比积分项(I)消除稳态误差微分项(D)抑制振荡参数作用典型值调整方向Kp响应速度20.0过大导致振荡Ki消除偏差0.5过大引起积分饱和Kd阻尼效果1.0抑制高频振动// 简易PID实现 typedef struct { float Kp, Ki, Kd; float integral; float prev_error; } PIDController; float pid_update(PIDController* pid, float error, float dt) { pid-integral error * dt; float derivative (error - pid-prev_error) / dt; pid-prev_error error; return pid-Kp*error pid-Ki*pid-integral pid-Kd*derivative; } // 使用示例 PIDController pid {20.0, 0.5, 1.0, 0, 0}; float output pid_update(pid, estimated_angle, 0.01); set_motor_speed(output);2.3 硬件集成要点传感器安装确保ADXL345的Y轴与小车前后方向一致电机驱动推荐使用TB6612FNG驱动模块电源管理单独为电机供电避免电压波动影响传感器机械结构重心位置决定控制难度注意纯加速度计方案在快速运动时会有明显延迟适合作为教学原型。实际产品建议搭配陀螺仪使用MPU6050等6轴IMU。3. 体感游戏控制器重新定义交互方式将ADXL345变身游戏控制器不仅是个酷炫的项目更能深入理解手势识别的基本原理。我们以控制贪吃蛇游戏为例展示如何将加速度数据转化为游戏指令。3.1 手势识别算法定义四个基本方向指令手势识别条件对应按键上抬Y轴持续0.8g达300ms方向上键下压Y轴持续-0.8g达300ms方向下键左倾X轴持续0.8g达300ms方向左键右倾X轴持续-0.8g达300ms方向右键// 手势状态跟踪结构体 typedef struct { float x_threshold; float y_threshold; uint32_t hold_duration; uint32_t start_time; bool detecting; } GestureDetector; bool detect_gesture(GestureDetector* det, float x, float y) { bool x_active fabs(x) det-x_threshold; bool y_active fabs(y) det-y_threshold; if(!det-detecting (x_active || y_active)) { det-detecting true; det-start_time HAL_GetTick(); return false; } if(det-detecting) { if((!x_active !y_active) || (HAL_GetTick() - det-start_time det-hold_duration)) { det-detecting false; return HAL_GetTick() - det-start_time det-hold_duration; } } return false; }3.2 数据无线传输通过蓝牙串口模块发送控制指令硬件连接STM32的USART接口连接HC-05协议设计简化指令格式如U\n表示向上抗干扰处理加入校验和与重传机制void send_gesture_command(char cmd) { static char last_cmd 0; if(cmd ! last_cmd) { uint8_t buf[3] {cmd, \n, 0}; HAL_UART_Transmit(huart1, buf, 2, 100); last_cmd cmd; } } // 在检测到手势后调用 if(detect_gesture(detector, acc_x, acc_y)) { if(acc_y 0.8) send_gesture_command(U); else if(acc_y -0.8) send_gesture_command(D); else if(acc_x 0.8) send_gesture_command(L); else if(acc_x -0.8) send_gesture_command(R); }3.3 性能优化技巧采样率匹配游戏通常30fps足够可降低传感器采样率省电运动去抖短时间内的连续手势只响应第一个校准功能长按按键进入校准模式自动计算中立位置电池供电利用STM32的低功耗模式延长使用时间4. 进阶技巧提升ADXL345应用性能当把这些项目真正做出来时你会发现几个共性的挑战。以下是经过多个项目验证的实用技巧。4.1 传感器校准方法论准确的测量从校准开始。ADXL345需要两种校准偏移校准将传感器静止放置在水平面记录100次采样取平均值计算各轴偏移量void calibrate_offset(int16_t* offset_x, int16_t* offset_y, int16_t* offset_z) { int32_t sum_x 0, sum_y 0, sum_z 0; for(int i0; i100; i) { int16_t x, y, z; adxl345_get_data(x, y, z); sum_x x; sum_y y; sum_z z; HAL_Delay(10); } *offset_x sum_x / 100; *offset_y sum_y / 100; *offset_z (sum_z / 100) - 256; // 假设Z轴正方向重力 }灵敏度校准分别将各轴对准重力方向测量输出值与理论值(256LSB/g)的比值4.2 数字滤波实战不同的应用需要不同的滤波策略滤波类型适用场景优缺点实现复杂度移动平均高频噪声简单有效但引入延迟★☆☆低通滤波平滑数据参数可调计算量适中★★☆卡尔曼滤波动态系统最优估计计算复杂★★★二阶低通滤波实现示例typedef struct { float a0, a1, a2, b1, b2; float x1, x2, y1, y2; } BiquadFilter; void init_lowpass(BiquadFilter* f, float cutoff, float sample_rate) { float omega 2 * M_PI * cutoff / sample_rate; float sn sin(omega); float cs cos(omega); float alpha sn / (2 * 0.7071); // Q0.7071 f-a0 1 alpha; f-a1 -2 * cs; f-a2 1 - alpha; f-b1 1 - cs; f-b2 (1 - alpha) / f-a0; f-a1 / f-a0; f-a2 / f-a0; f-b1 / f-a0; f-b2 / f-a0; } float biquad_process(BiquadFilter* f, float x) { float y x - f-a1*f-x1 - f-a2*f-x2 f-b1*f-y1 f-b2*f-y2; f-x2 f-x1; f-x1 x; f-y2 f-y1; f-y1 y; return y; }4.3 资源优化策略在STM32F103这类资源有限的MCU上合理配置数据速率平衡响应速度与噪声使用定点数运算Q格式数学比浮点高效选择性更新只有变化超过阈值时才处理数据DMA传输解放CPU处理其他任务// 使用Q15定点数示例 int16_t accel_to_q15(float g) { return (int16_t)(g * 32767 / 16); // ±16g范围映射到Q15 } float q15_to_accel(int16_t q) { return (float)q * 16 / 32767; }在完成这些项目后你会发现ADXL345这个看似简单的传感器其实蕴含着改变物理世界交互方式的巨大能量。从保护生命的严肃应用到充满乐趣的游戏创新正是这种跨越领域的可能性让嵌入式开发如此迷人。

更多文章