STM32F103C8T6 HC-SR04超声波模块——超声波障碍物测距(HAl库)

2024-05-25 16:04

本文主要是介绍STM32F103C8T6 HC-SR04超声波模块——超声波障碍物测距(HAl库),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

超声波障碍物测距

  • 一、HC-SR04超声波模块
    • (一)什么是HC-SR04?
    • (二)HC-SR04工作原理
    • (三)如何使用HC-SR04
    • (四)注意事项
  • 二、程序编写
    • (一)CubeMX配置
      • 1.芯片选择
      • 2.配置RCC、SYS、时钟树![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/8e04c33ec7fe4fdbbd1a9ef7593e74d8.png)
      • 3.配置GPIO
      • 4.配置串口1
      • 5.配置定时器
      • 6.开启定时器中断
      • 7.设置路径、生成代码工程
    • (二)代码编写
  • 三、效果展示
  • 四、总结
  • 五、引用

使用的软硬件设备与应用:

  • 硬件:STM32F103C8T6、HC-SR04、LED灯
  • 软件:Keil5、CubeMX、串口助手

一、HC-SR04超声波模块

(一)什么是HC-SR04?

HC-SR04是一款常用的超声波测距模块,它能够通过发送超声波脉冲并接收其回波来测量物体与传感器之间的距离。这种模块广泛应用于机器人、智能小车、智能家居等领域,可以用于测量墙壁距离、障碍物距离、物品距离。

(二)HC-SR04工作原理

HC-SR04模块的工作原理是,通过IO口TRIG触发测距,给至少10us的高电平信号,模块自动发送8个40kHz的方波,并自动检测是否有信号返回。如果有信号返回,通过IO口ECHO输出一个高电平,高电平持续的时间即为超声波从发射到返回的时间。通过测量这个时间,可以计算出距离,公式为:测试距离 = (高电平时间 × 声速(340m/s)) / 2。

(三)如何使用HC-SR04

模块的电气连接通常包括四个引脚:VCC、Trig、Echo、GND。VCC和GND分别为正负电源引脚,Trig为触发引脚,Echo为回波引脚。在使用时,需要给Trig引脚发送一个10us以上的高电平信号,然后在Echo引脚等待高电平输出,当Echo引脚变为低电平时,读取定时器的值,即为此次测距的时间。

(四)注意事项

  • 不宜带电连接模块,如果要带电连接,则先让模块的GND端先连接,否则会影响模块工作。
  • 测距时,被测物体的面积不少于0.5平方米且要尽量平整,否则会影响测试结果。
  • HC-SR04模块的典型工作电压为5V,静态工作电流小于2mA,感应角度不大于15度,探测距离为2cm-400cm,精度可达0.3cm,存在一个2cm的盲区。2020版本的HC-SR04支持GPIO、UART和IIC三种模式接口,工作电压3-5.5V,测量范围2cm-450cm。

二、程序编写

(一)CubeMX配置

1.芯片选择

在这里插入图片描述

2.配置RCC、SYS、时钟树在这里插入图片描述

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

3.配置GPIO

A1连接的是ECHO,B9为LED
在这里插入图片描述

4.配置串口1

在这里插入图片描述

5.配置定时器

在这里插入图片描述

6.开启定时器中断

在这里插入图片描述

7.设置路径、生成代码工程

在这里插入图片描述

(二)代码编写

main.c:

/* USER CODE BEGIN Header */
/********************************************************************************* @file           : main.c* @brief          : Main program body******************************************************************************* @attention** <h2><center>&copy; Copyright (c) 2022 STMicroelectronics.* All rights reserved.</center></h2>** This software component is licensed by ST under BSD 3-Clause license,* the "License"; You may not use this file except in compliance with the* License. You may obtain a copy of the License at:*                        opensource.org/licenses/BSD-3-Clause********************************************************************************/
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "tim.h"
#include "usart.h"
#include "gpio.h"/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "SR04.h"
#include "led.h"
/* USER CODE END Includes *//* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD *//* USER CODE END PTD *//* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD *//* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM *//* USER CODE END PM *//* Private variables ---------------------------------------------------------*//* USER CODE BEGIN PV *//* USER CODE END PV *//* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP *//* USER CODE END PFP *//* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 *//* USER CODE END 0 *//*** @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_TIM2_Init();MX_USART1_UART_Init();/* USER CODE BEGIN 2 *//* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */while (1){float distance = SR04_GetData();//HAL_Delay(1500);/* USER CODE END WHILE */// 根据距离计算闪烁频率uint32_t flashRate = CalculateFlashRate(distance);LED_Flash(flashRate); // 闪烁LED/* USER CODE BEGIN 3 */}/* USER CODE END 3 */
}/*** @brief System Clock Configuration* @retval None*/
void SystemClock_Config(void)
{RCC_OscInitTypeDef RCC_OscInitStruct = {0};RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};/** Initializes the RCC Oscillators according to the specified parameters* in the RCC_OscInitTypeDef structure.*/RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;RCC_OscInitStruct.HSEState = RCC_HSE_ON;RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;RCC_OscInitStruct.HSIState = RCC_HSI_ON;RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK){Error_Handler();}/** Initializes the CPU, AHB and APB buses clocks*/RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK){Error_Handler();}
}/* USER CODE BEGIN 4 *//* USER CODE END 4 *//*** @brief  This function is executed in case of error occurrence.* @retval None*/
void Error_Handler(void)
{/* USER CODE BEGIN Error_Handler_Debug *//* User can add his own implementation to report the HAL error return state */__disable_irq();while (1){}/* USER CODE END Error_Handler_Debug */
}#ifdef  USE_FULL_ASSERT
/*** @brief  Reports the name of the source file and the source line number*         where the assert_param error has occurred.* @param  file: pointer to the source file name* @param  line: assert_param error line source number* @retval None*/
void assert_failed(uint8_t *file, uint32_t line)
{/* USER CODE BEGIN 6 *//* User can add his own implementation to report the file name and line number,ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) *//* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */

SR04.h:

#ifndef __SR04_H
#define __SR04_H
#include "main.h"
#include "tim.h"
#include "stdio.h"#define TRIG_H  HAL_GPIO_WritePin(Trig_GPIO_Port,Trig_Pin,GPIO_PIN_SET)
#define TRIG_L  HAL_GPIO_WritePin(Trig_GPIO_Port,Trig_Pin,GPIO_PIN_RESET)void delay_us(uint32_t us);
float SR04_GetData(void);#endif

SR04.c:

#include "SR04.h"
#include "stm32f1xx_hal.h" float distant;      //测量距离
uint32_t measure_Buf[3] = {0};   //存放定时器计数值的数组
uint8_t  measure_Cnt = 0;    //状态标志位
uint32_t high_time;   //超声波模块返回的高电平时间//===============================================读取距离
float SR04_GetData(void)
{switch (measure_Cnt){case 0:TRIG_H;delay_us(30);TRIG_L;measure_Cnt++;__HAL_TIM_SET_CAPTUREPOLARITY(&htim2, TIM_CHANNEL_1, TIM_INPUTCHANNELPOLARITY_RISING);HAL_TIM_IC_Start_IT(&htim2, TIM_CHANNEL_1); // 启动输入捕获break;case 3:high_time = measure_Buf[1] - measure_Buf[0]; // 高电平时间printf("\r\n----高电平时间-%d-us----\r\n", high_time);float distance = (high_time * 0.034f) / 2; // 单位cmprintf("\r\n-检测距离为-%.2f-cm-\r\n", distance);measure_Cnt = 0; // 清空标志位TIM2->CNT = 0; // 清空计时器计数// 返回计算得到的距离值return distance;}return 0; // 如果没有测量完成,返回0或合适的默认值
}//===============================================us延时函数void delay_us(uint32_t us)//主频72M
{uint32_t delay = (HAL_RCC_GetHCLKFreq() / 4000000 * us);while (delay--){;}
}//===============================================中断回调函数
void HAL_TIM_IC_CaptureCallback(TIM_HandleTypeDef *htim)//
{if(TIM2 == htim->Instance)// 判断触发的中断的定时器为TIM2{switch(measure_Cnt){case 1:measure_Buf[0] = HAL_TIM_ReadCapturedValue(&htim2,TIM_CHANNEL_1);//获取当前的捕获值.__HAL_TIM_SET_CAPTUREPOLARITY(&htim2,TIM_CHANNEL_1,TIM_ICPOLARITY_FALLING);  //设置为下降沿捕获measure_Cnt++;                                            break;              case 2:measure_Buf[1] = HAL_TIM_ReadCapturedValue(&htim2,TIM_CHANNEL_1);//获取当前的捕获值.HAL_TIM_IC_Stop_IT(&htim2,TIM_CHANNEL_1); //停止捕获   或者: __HAL_TIM_DISABLE(&htim5);measure_Cnt++;  }}}

led.h:

#ifndef __LED_H__
#define __LED_H__#include "stm32f1xx_hal.h"void LED_Init(void);
void LED_Flash(uint32_t period); 
uint32_t CalculateFlashRate(float distance);#endif // __LED_H__

led.c:

#include "led.h"#define LED_GPIO_Port       GPIOB
#define LED_Pin             GPIO_PIN_9void LED_Init(void) {GPIO_InitTypeDef GPIO_InitStruct = {0};// 使能GPIOB时钟__HAL_RCC_GPIOB_CLK_ENABLE();// 配置GPIO Pin为推挽输出,无上拉电阻,低频率GPIO_InitStruct.Pin = LED_Pin;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init(LED_GPIO_Port, &GPIO_InitStruct);
}void LED_Flash(uint32_t period) {// 切换LED状态HAL_GPIO_TogglePin(LED_GPIO_Port, LED_Pin);// 延时HAL_Delay(period);// 再次切换LED状态,完成一次闪烁HAL_GPIO_TogglePin(LED_GPIO_Port, LED_Pin);
}uint32_t CalculateFlashRate(float distance) {uint32_t flashRate;if (distance < 10.0f) {flashRate = 100; // 距离小于10厘米时,快速闪烁} else if (distance < 50.0f) {flashRate = 500; // 距离在10-50厘米之间时,中速闪烁} else {flashRate = 1000; // 距离大于50厘米时,慢速闪烁}return flashRate;
}

三、效果展示

在这里插入图片描述

HC-SR04

四、总结

本次实现了STM32F103C8T6微控制器和HC-SR04超声波测距模块的结合应用,并通过LED闪烁频率表现出来。通过CubeMX的图形化配置和Keil5的开发环境,项目简化了开发流程,提高了开发效率。此外,项目还提供了对超声波测距原理的深入理解,以及如何通过软件控制硬件来实现特定功能的方法。

五、引用

【嵌入式创客工坊】STM32系列(HAL库)——F103C8T6通过HC-SR04超声波模块实现测距

这篇关于STM32F103C8T6 HC-SR04超声波模块——超声波障碍物测距(HAl库)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1001968

相关文章

Python logging模块使用示例详解

《Pythonlogging模块使用示例详解》Python的logging模块是一个灵活且强大的日志记录工具,广泛应用于应用程序的调试、运行监控和问题排查,下面给大家介绍Pythonlogging模... 目录一、为什么使用 logging 模块?二、核心组件三、日志级别四、基本使用步骤五、快速配置(bas

Python datetime 模块概述及应用场景

《Pythondatetime模块概述及应用场景》Python的datetime模块是标准库中用于处理日期和时间的核心模块,本文给大家介绍Pythondatetime模块概述及应用场景,感兴趣的朋... 目录一、python datetime 模块概述二、datetime 模块核心类解析三、日期时间格式化与

Python如何调用指定路径的模块

《Python如何调用指定路径的模块》要在Python中调用指定路径的模块,可以使用sys.path.append,importlib.util.spec_from_file_location和exe... 目录一、sys.path.append() 方法1. 方法简介2. 使用示例3. 注意事项二、imp

Python中模块graphviz使用入门

《Python中模块graphviz使用入门》graphviz是一个用于创建和操作图形的Python库,本文主要介绍了Python中模块graphviz使用入门,具有一定的参考价值,感兴趣的可以了解一... 目录1.安装2. 基本用法2.1 输出图像格式2.2 图像style设置2.3 属性2.4 子图和聚

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

Python中的getopt模块用法小结

《Python中的getopt模块用法小结》getopt.getopt()函数是Python中用于解析命令行参数的标准库函数,该函数可以从命令行中提取选项和参数,并对它们进行处理,本文详细介绍了Pyt... 目录getopt模块介绍getopt.getopt函数的介绍getopt模块的常用用法getopt模

python logging模块详解及其日志定时清理方式

《pythonlogging模块详解及其日志定时清理方式》:本文主要介绍pythonlogging模块详解及其日志定时清理方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录python logging模块及日志定时清理1.创建logger对象2.logging.basicCo

Qt spdlog日志模块的使用详解

《Qtspdlog日志模块的使用详解》在Qt应用程序开发中,良好的日志系统至关重要,本文将介绍如何使用spdlog1.5.0创建满足以下要求的日志系统,感兴趣的朋友一起看看吧... 目录版本摘要例子logmanager.cpp文件main.cpp文件版本spdlog版本:1.5.0采用1.5.0版本主要

Python使用date模块进行日期处理的终极指南

《Python使用date模块进行日期处理的终极指南》在处理与时间相关的数据时,Python的date模块是开发者最趁手的工具之一,本文将用通俗的语言,结合真实案例,带您掌握date模块的六大核心功能... 目录引言一、date模块的核心功能1.1 日期表示1.2 日期计算1.3 日期比较二、六大常用方法详