基于STM32的智能超声波测距与多级报警系统开发(附仿真与源码)
1. 项目背景与核心功能
超声波测距技术在现代智能设备中的应用越来越广泛,从智能家居到工业自动化都能看到它的身影。这次我们要做的项目,是用STM32单片机搭配HC-SR04超声波传感器,打造一个带有多级报警功能的测距系统。这个系统不仅能实时测量4cm到250cm范围内的距离,还能通过LED、蜂鸣器和OLED显示屏实现三级报警提示。
我最早接触这个项目是在帮朋友改造车库门禁系统时。当时需要检测车辆与门之间的距离,试过红外和激光方案后,发现超声波在成本和抗干扰性上表现更均衡。实测下来,HC-SR04在3米内的测距误差能控制在±2cm以内,完全满足大多数场景需求。
系统最实用的功能是动态阈值报警。比如用在智能停车系统时,当车辆接近障碍物到1米距离,OLED会显示黄色警告;距离小于50cm时,蜂鸣器开始间歇报警;若继续接近到30cm内,红色LED会常亮并触发急促警报声。这三个级别的报警阈值都可以通过按键随时调整,适应不同应用场景。
2. 硬件设计详解
2.1 核心控制器选型
STM32F103C8T6是这个项目的核心,我选它的原因很实际:价格便宜(某宝10元左右)、性能足够(72MHz主频)、外设丰富。特别是它的定时器功能,用来捕获超声波回波信号的高电平时间非常方便。记得第一次调试时用Arduino做,虽然开发简单但实时性不够,后来换STM32的定时器输入捕获模式,测量稳定性立刻提升不少。
硬件连接时要注意:
- VCC接3.3V或5V(HC-SR04兼容两种电压)
- Trig接PA1(任意GPIO均可)
- Echo最好接具有输入捕获功能的引脚如PA0
// STM32引脚配置示例 GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); // Trig引脚配置为推挽输出 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); // Echo引脚配置为浮空输入 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(GPIOA, &GPIO_InitStructure);2.2 超声波传感器工作原理
HC-SR04的工作流程其实很有意思。当你给Trig引脚一个10μs以上的高电平脉冲后,模块会自动发射8个40kHz的超声波脉冲,然后Echo引脚会输出高电平,这个高电平的持续时间就是超声波从发射到返回的时间。
计算距离的公式很简单:
距离(cm) = (高电平时间(μs) × 声速(340m/s)) / 2 × 10000实际编程时需要做温度补偿,因为声速会随温度变化。我在代码中添加了DS18B20温度传感器的支持,使测距精度提升了约15%。
2.3 报警模块设计
多级报警是这个项目的亮点:
- 一级报警(阈值距离的80%):OLED显示黄色警告
- 二级报警(阈值距离的50%):蜂鸣器间歇鸣响(500ms间隔)
- 三级报警(阈值距离的30%):红色LED常亮 + 蜂鸣器连续报警
蜂鸣器驱动电路有个小技巧:在STM32和蜂鸣器之间加个三极管(如S8050),用PWM控制可以调节音量大小。我曾经直接用IO口驱动,结果声音小得几乎听不见。
// 报警控制代码示例 void Alarm_Control(uint16_t distance) { if(distance < threshold * 0.3) { // 三级报警 GPIO_SetBits(LED_PORT, LED_PIN); PWM_SetDuty(BUZZER_PWM, 90); // 90%占空比 } else if(distance < threshold * 0.5) { // 二级报警 PWM_SetDuty(BUZZER_PWM, 70); GPIO_ResetBits(LED_PORT, LED_PIN); } else if(distance < threshold) { // 一级报警 OLED_ShowWarning(); } }3. 软件实现关键点
3.1 测距算法优化
原始的实现直接使用定时器捕获Echo高电平时间,但在实际测试中发现两个问题:
- 远距离测量时容易受环境噪声干扰
- 多次测量结果会有±3cm左右的跳动
改进后的算法做了三件事:
- 连续采样5次取中值:过滤突发干扰
- 动态调整采样间隔:近距离时加快采样(100ms),远距离时减慢(300ms)
- 移动平均滤波:对最后3次有效结果求平均
// 改进后的测距函数 float Get_Distance(void) { uint32_t buffer[5]; for(int i=0; i<5; i++) { buffer[i] = HC_SR04_GetPulse(); delay_ms(20); } // 排序取中值 Bubble_Sort(buffer, 5); float distance = (buffer[2] * 0.0343) / 2; return Moving_Average(distance); // 移动平均 }3.2 多级阈值管理
阈值管理我设计了两种模式:
- 固定模式:通过按键设置静态阈值(适合固定场景)
- 动态模式:系统自动学习环境变化(适合复杂场景)
动态模式的实现比较有意思:系统会记录最近100次测量结果,当检测到持续10次以上距离小于当前阈值的80%时,自动下调阈值;反之则上调。这个功能在智能车库项目中特别实用,能自动适应不同尺寸的车辆。
// 动态阈值调整算法 void Adjust_Threshold(float current_dist) { static float history[100]; static int index = 0; history[index++] = current_dist; if(index >= 100) index = 0; // 计算最近10次是否持续低于阈值 int count = 0; for(int i=0; i<10; i++) { int pos = (index - i - 1 + 100) % 100; if(history[pos] < threshold * 0.8) count++; } if(count >= 8) threshold *= 0.95; // 下调5% else if(count <=2) threshold *= 1.05; // 上调5% }3.3 OLED界面设计
使用u8g2库驱动OLED显示,界面布局遵循三个原则:
- 关键信息突出:实时距离用24号字体
- 状态可视化:用颜色和图标表示报警级别
- 操作反馈明确:调整阈值时有进度条动画
// OLED显示示例 void OLED_Refresh(void) { u8g2_ClearBuffer(&u8g2); u8g2_SetFont(&u8g2, u8g2_font_wqy16_t_gb2312); u8g2_DrawUTF8(&u8g2, 0, 16, "当前距离:"); u8g2_SetFont(&u8g2, u8g2_font_logisoso24_tf); char buf[10]; sprintf(buf, "%.1fcm", distance); u8g2_DrawStr(&u8g2, 30, 45, buf); // 绘制阈值进度条 u8g2_DrawFrame(&u8g2, 0, 50, 128, 10); u8g2_DrawBox(&u8g2, 0, 50, (int)(128*threshold/250), 10); u8g2_SendBuffer(&u8g2); }4. Proteus仿真与调试技巧
4.1 仿真环境搭建
在Proteus中搭建仿真环境时,需要注意几个关键点:
- HC-SR04模型参数:将Echo Response Time设置为与实际相符的1ms-25ms
- STM32时钟配置:在Project Settings中设置正确的晶振频率(通常8MHz)
- 虚拟终端:连接USART1用于调试输出
仿真时发现一个有趣现象:Proteus的超声波传感器模型对Echo信号有最小脉宽限制(约1ms),这意味着仿真中无法测试30cm以内的近距离测量。后来我修改了模型参数文件才解决这个问题。
4.2 常见问题排查
在项目开发过程中,我踩过不少坑,这里分享三个典型问题的解决方法:
问题1:测量结果跳动大
- 检查电源稳定性(最好给HC-SR04单独供电)
- 在Trig和Echo线上加10kΩ上拉电阻
- 确保测量期间没有其他中断干扰
问题2:远距离测量不准
- 调整传感器朝向(被测物体表面要平整)
- 在代码中增加超时判断(超过38ms视为无效)
- 添加温度补偿(声速=331.4 + 0.6×温度℃)
问题3:蜂鸣器不响
- 检查驱动电路三极管是否接反
- 用示波器确认PWM信号输出
- 尝试降低蜂鸣器工作电压(有些5V蜂鸣器在3.3V下不响)
// 带温度补偿的测距代码 float Get_Distance_With_Temp(float temperature) { float sound_speed = 331.4 + 0.6 * temperature; uint32_t pulse = HC_SR04_GetPulse(); return (pulse * 1e-6 * sound_speed * 100) / 2; }5. 进阶功能扩展
5.1 无线传输模块
给系统添加ESP8266 WiFi模块后,可以实现距离数据的远程监控。我在一个智能仓储项目中就用了这个方案,通过MQTT协议将各个节点的测距数据上传到服务器,实现立体仓库的实时监控。
接线很简单:
- ESP8266的TX接STM32的PA3(USART2_RX)
- ESP8266的RX接STM32的PA2(USART2_TX)
- 共地连接
// WiFi数据上传示例 void WiFi_SendData(float dist) { char msg[64]; sprintf(msg, "{\"id\":%d,\"distance\":%.1f}", device_id, dist); USART2_SendString("AT+CIPSEND=0,"); USART2_SendInt(strlen(msg)); USART2_SendString("\r\n"); delay_ms(100); USART2_SendString(msg); }5.2 多传感器阵列
对于需要广角检测的场景,可以用多个HC-SR04组成传感器阵列。我设计过一个五向测距模块,用STM32的定时器复用功能同时管理5个传感器:
- 使用74HC138译码器轮流触发不同传感器
- 所有Echo信号通过或门合并到一个中断引脚
- 在中断中读取当前激活的传感器编号
这种设计只需要占用4个GPIO(3个用于译码器输入,1个中断引脚),却可以扩展最多8个超声波传感器。
5.3 能耗优化技巧
对于电池供电的应用,我总结了几个省电技巧:
- 间歇工作模式:每5秒唤醒一次,测量后立即休眠
- 动态功率控制:远距离测量时提高发射功率
- 关闭非必要外设:不用OLED时关闭其电源
- 降低主频:在休眠期间将系统时钟切换到HSI 8MHz
// 低功耗模式配置 void Enter_LowPowerMode(void) { RCC_SYSCLKConfig(RCC_SYSCLKSource_HSI); // 切换到内部8MHz时钟 GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = OLED_PWR_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(OLED_PWR_PORT, &GPIO_InitStructure); GPIO_ResetBits(OLED_PWR_PORT, OLED_PWR_PIN); // 关闭OLED电源 PWR_EnterSTOPMode(PWR_Regulator_LowPower, PWR_STOPEntry_WFI); }6. 实际应用案例
6.1 智能车库防撞系统
这是我给朋友改装的车库项目,主要解决他倒车时经常蹭到后墙的问题。系统安装在车库后墙,当检测到车辆距离小于30cm时,会通过LED灯带发出红色光幕警告,同时车载收音机会自动切换到一个特定频率播放语音提示。
关键改进点:
- 增加了防水外壳(3D打印的ABS盒子)
- 使用高亮度LED灯带替代普通LED
- 添加了RF433模块无线触发车载设备
6.2 工业传送带堵料检测
在某食品厂的包装产线上,我们部署了这个系统的改进版用于检测传送带上的物料堆积。当两个相邻传感器同时检测到距离持续小于设定值超过5秒时,系统会自动暂停传送带并触发声光报警。
工业环境的特殊处理:
- 传感器加装金属防护罩
- 所有信号线改用屏蔽双绞线
- 增加RS485总线通信
- 报警阈值设置为可远程修改
6.3 智能家居安防应用
将系统安装在窗户附近,可以检测是否有人靠近。与家居自动化平台(如Home Assistant)联动后,可以实现以下场景:
- 检测到有人靠近时自动开灯
- 持续靠近超过阈值时拍照并推送警报
- 与窗帘电机联动实现自动关闭
这个版本增加了PIR人体传感器作为双重验证,大幅降低了误报率。
