STM32f407驱动hc-sr04超声波测距模块

本文介绍了如何使用STM32f407驱动HC-SR04超声波测距模块,详细阐述了模块的工作原理、供电需求以及操作步骤。在实践中,作者提醒读者避免直接复制网上质量不高的代码,而是要理解并自行计算。文中提到了计算距离的公式,并给出了最大测量距离4米时的定时器设置。最后,分享了实现该功能的代码。
摘要由CSDN通过智能技术生成

最近完了一下测距模块,本想挺简单的一个东西,却折腾了好长时间。总算弄好了,贴到这里和大家分享一下!

HC-SR04超声波测距模块是一种被用烂了的传感器,它的优点是成本低,控制容易,但是掌握不好时序,还是会弄的人一头雾水。

本模块用使用方法如下:

一.  5v直流电压供电,一般的单片机上自带的VCC都能驱动

二.  Trig引脚收到来自单片机时长大于等于20us的高点品后发出超声波

三.  Echo引脚接收Trig引脚发出的超声波的回波,并且单片机开启定时器记录高电平时间

四. 高电平结束后,关闭定时器,从定时器寄存器中取出记录值,换算成距离即可。

在做的过程中注意:

网上有好多代码,但是大多数互相抄袭,而且代码质量都不怎么高,每一句话什么意思也不明白,所以不要照抄网上那些代码,还是要自己琢磨;

在换算距离时,网上的代码各种各样,但结果都不符合实际,于是自己算了一下:

在我的程序中,定时器计时频率是200000,周期就是1/200000,在这个周期里,超声波走的距离是(1/200000)*344/2=0.00085米,换算成毫米是0.86米,精度符合要求。

在获取定时器的计数器值TIM,于是TIM*0.85就是实际距离,单位是毫米。

另外,超声波测距模块的最远测量距离是4米,可以算出距离在4米事定时器的计数值,x*340=4,得x=5000,所以设置定时器的溢出周一为5000.

好了,贴上代码吧!<

以下是使用STM32F407HC-SR04超声波测距模块的示例代码: ```c #include "stm32f4xx.h" #include "stm32f4xx_gpio.h" #include "stm32f4xx_rcc.h" #include "stm32f4xx_tim.h" #include "misc.h" #define TRIG_PIN GPIO_Pin_0 #define ECHO_PIN GPIO_Pin_1 #define GPIO_PORT GPIOA uint32_t distance = 0; void GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = TRIG_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(GPIO_PORT, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = ECHO_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(GPIO_PORT, &GPIO_InitStructure); } void TIM_Configuration(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); TIM_TimeBaseStructure.TIM_Period = 0xFFFFFFFF; TIM_TimeBaseStructure.TIM_Prescaler = (uint16_t)(SystemCoreClock / 1000000) - 1; TIM_TimeBaseStructure.TIM_ClockDivision = 0; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); TIM_ICInitTypeDef TIM_ICInitStructure; TIM_ICInitStructure.TIM_Channel = TIM_Channel_2; TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; TIM_ICInitStructure.TIM_ICFilter = 0x0; TIM_ICInit(TIM2, &TIM_ICInitStructure); TIM_Cmd(TIM2, ENABLE); } uint32_t GetDistance(void) { GPIO_SetBits(GPIO_PORT, TRIG_PIN); Delay_us(10); GPIO_ResetBits(GPIO_PORT, TRIG_PIN); uint32_t start_time = 0; uint32_t end_time = 0; uint32_t timeout = 0; while (GPIO_ReadInputDataBit(GPIO_PORT, ECHO_PIN) == RESET) { if (++timeout > 1000000) { return 0xFFFFFFFF; } } start_time = TIM_GetCounter(TIM2); timeout = 0; while (GPIO_ReadInputDataBit(GPIO_PORT, ECHO_PIN) == SET) { if (++timeout > 1000000) { return 0xFFFFFFFF; } } end_time = TIM_GetCounter(TIM2); if (end_time > start_time) { distance = (end_time - start_time) * 10 / 58; } else { distance = (start_time - end_time) * 10 / 58; } return distance; } int main(void) { GPIO_Configuration(); TIM_Configuration(); while (1) { GetDistance(); } return 0; } ``` 此代码初始化GPIO和定时器,然后在主循环中调用`GetDistance()`函数来获取测量到的距离。该函数使用超声波模块发送一个10微秒的脉冲,然后等待接收到回波。通过测量回波的时间来计算距离,并返回结果。在本例中,距离以毫米为单位表示。如果无法检测到回波或测量超时,则返回0xFFFFFFFF。 需要注意的是,在本例中,我们使用了定时器的输入捕获模式来测量回波的时间。我们使用TIM2的通道2来读取ECHO引脚的状态,并在上升沿触发捕获。此代码假设您已正确配置STM32F407的时钟源。如果您遇到问题,请参阅STM32F407参考手册。
评论 18
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值