基于STM32的HC_SR04模块实现超声波测距(附源码)

一、实验简介及原理

本次实验需要通过STM32与HC_SR04模块实现实时测距,并将测距信息通过串口显示在电脑上

原理

超声波测距原理是在超声波发射装置发出超声波,它的根据是接收器接到超声波时的时间差,与雷达测距原理相似。 超声波发射器向某一方向发射超声波,在发射时刻的同时开始计时,超声波在空气中传播,途中碰到障碍物就立即返回来,超声波接收器收到反射波就立即停止计时。
超声波在空气中的传播速度为340m/s,根据计时器记录的时间t(秒),就可以计算出发射点距障碍物的距离(s),即:s=340t/2

HC-SR04模块

  • 采用IO口TRIG触发测距,给至少10us的高电平信号,模块自动发送8个40khz的方波,自动检测是否有信号返回。
  • 有信号返回,通过IO口ECHO输出一个高电平,高电平持续的时间就是超声波从发射到返回的时间。测试距离=(高电平时间*声速(340M/S))/2。
    在这里插入图片描述

二、实验设备

硬件:STM32F103C8T6、HC_SR04超声波测距模块
软件:CubeMX、Keil5、FlyMCU、野火串口调试助手

三、实验主要代码

(完整源码在代码块后)

uint32_t csb_get_distance(void)
{
    uint32_t CSB_value = 0 ;
	//给发射引脚一个高电平
	HAL_GPIO_WritePin(Trig_GPIO_Port, Trig_Pin, GPIO_PIN_SET);
	//延时10us以上
	TIM1_Delay_us(20);
	//给发射引脚一个低电平
	HAL_GPIO_WritePin(Echo_GPIO_Port, Trig_Pin, GPIO_PIN_RESET);
	//等待接收引脚变为高电平
	while( HAL_GPIO_ReadPin(Echo_GPIO_Port,Echo_Pin) == 0);
	//设置定时器初始值为0
	__HAL_TIM_SetCounter(&htim1, 0);
	//开始计时
	__HAL_TIM_ENABLE(&htim1);
	//当接受引脚从高电平转换为低电平时,停止计时,获取计时时间
    while( HAL_GPIO_ReadPin(Echo_GPIO_Port,Echo_Pin) == 1);  
	//获取定时器的计数值a = b;
	CSB_value = __HAL_TIM_GetCounter(&htim1);
	//停止计时
	__HAL_TIM_DISABLE(&htim1);
	//已知高电平总时间,即可利用公式测试距离= (高电平时间*声速(340M/S))/2
	//如果需要返回毫米级别距离,公式为:测试距离= (高电平时间*声逮(340M/1000ms))/2
	return ( CSB_value*340/1000/2);
}
/* USER CODE END 0 */
uint32_t data[5] = {0};
uint32_t averageFilter(uint32_t in_data)
{
	float sum = 0;
	for(int i=0; i<4; i++)
	{
		data[i]=data[i+1];
		sum = sum + data[i];
	}
	data[4] = in_data;
	sum = sum + data[4];
	
	return(sum/5);
}
/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_TIM1_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
	int distance;
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
		distance = averageFilter(csb_get_distance());	
		if(distance<=10)
		{
			HAL_GPIO_WritePin(led_GPIO_Port, led_Pin, GPIO_PIN_RESET);
		}
		else
			HAL_GPIO_WritePin(led_GPIO_Port, led_Pin, GPIO_PIN_SET);
		printf("超声波检测距离%dcm\n",	distance/10);
		HAL_Delay(500);
    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

完整源码:
链接:https://pan.baidu.com/s/1YR_0m4aP8GznQenHikV3ug
提取码:n3mq

四、最终结果

在这里插入图片描述
在这里插入图片描述

  • 7
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是基于STM32hc-sr04超声波测距模块代码示例: ```c #include "stm32f10x.h" #define Trig_Pin GPIO_Pin_0 #define Echo_Pin GPIO_Pin_1 #define Trig_Port GPIOC #define Echo_Port GPIOC void TIM2_Configuration(void); float ultrasonic_measure(void); int main(void) { float distance; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE); GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.GPIO_Pin = Trig_Pin; GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(Trig_Port, &GPIO_InitStruct); GPIO_InitStruct.GPIO_Pin = Echo_Pin; GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(Echo_Port, &GPIO_InitStruct); TIM2_Configuration(); while(1) { distance = ultrasonic_measure(); } } void TIM2_Configuration(void) { RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct; TIM_TimeBaseInitStruct.TIM_Prescaler = 72 - 1; TIM_TimeBaseInitStruct.TIM_Period = 0xFFFF; TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseInitStruct); TIM_OCInitTypeDef TIM_OCInitStruct; TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OCInitStruct.TIM_Pulse = 0; TIM_OC1Init(TIM2, &TIM_OCInitStruct); TIM_Cmd(TIM2, ENABLE); } float ultrasonic_measure(void) { uint16_t TIM2_CountStart, TIM2_CountEnd; float Distance; GPIO_ResetBits(Trig_Port, Trig_Pin); TIM_SetCounter(TIM2, 0); while(TIM_GetCounter(TIM2) < 10); // delay 10us GPIO_SetBits(Trig_Port, Trig_Pin); while(GPIO_ReadInputDataBit(Echo_Port, Echo_Pin) == RESET); TIM_Cmd(TIM2, ENABLE); while(GPIO_ReadInputDataBit(Echo_Port, Echo_Pin) == SET); TIM_Cmd(TIM2, DISABLE); TIM2_CountStart = TIM2->CCR1; TIM2_CountEnd = TIM_GetCapture1(TIM2); Distance = (float)(TIM2_CountEnd - TIM2_CountStart) * 0.01715f; // Distance = (High level time * sound velocity (340M/S) / 2) return Distance; } ``` 代码中使用了STM32的定时器TIM2进行计时,测量距离的原理是通过发送一个10us的高电平脉冲触发hc-sr04模块,然后接收返回的超声波信号并测量高电平持续时间,最后通过公式计算出距离。需要注意的是,本示例中使用的是72MHz的系统时钟,如果系统时钟不同,需要相应地修改计时器的预分频值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值