STM32——HAL库开发笔记24(定时器5—超声波测距)(参考来源:b站铁头山羊)

一、原理

本次实验采用HC-SR04超声波传感器,结构及功能如下图

超声波传感器可以用来测距。距离 = 声速(340m/s) *传播时间 / 2。

这个传感器有四个引脚,其中VCC接电源正极,GND接电源负极 , Trig : 用来启动测量  ,Echo用来返回结果。

在开始测量之前,我们要给Trig引脚施加一个大于10微秒的脉冲,然后T开始发送超声波,整个过程大约持续0.2毫秒,声波发送结束后Echo引脚上会出现一个上升沿,然后当8个周期 的超声波全部接收到后,Echo引脚上又会出现一个下降沿。这样在Echo引脚上就形成了一个脉冲,我们测量脉冲的宽度就能得到超声波在空气中传播的时间。

二、STM32CubeMX配置

1:设置调试接口

2:设置PA0

3:设置定时器——通道一选择上升沿直接,通道二选择下降沿间接

4:配置板载LED:PC13引脚

 接下来用几张图回顾一下定时器的工作过程

分辨率:

所以

5:设置定时器参数

6:保存并且打开工程

三、电路连接图

四、编程接口

HAL_StatusTypeDef HAL_TIM_IC_Start(TIM_HandleTypeDef *htim , uint32_t Channel)

作用 :启动输入捕获


HAL_StatusTypeDef HAL_TIM_IC_Stop(TIM_HandleTypeDef *htim , uint32_t Channel)

作用 :停止输入捕获

参数说明:htim :定时器句柄指针
         Channel  : 通道编号

五、编程思路

1:让CNT的值归0

2:清除CCR1和CCR2标志位

CNT溢出时,会触发update,当update发生的时候,单片机会自动让update标志位从0到1;所以我们只需要查询一下update的标志位就可以知道CNT有没有溢出。当我们把定时器通道设置为输入捕获模式,当这些通道捕捉到了对应的信号变化时就会产生一个CCx时间,单片机会自动的将CCx标志位从0变到1,所以我们只需要查询一下CCx标注位就可以知道这个通道当前有没有捕捉到信号变化。在使用之前需要对其清0。

3:启动定时器

4:向Trig引脚发送脉冲:

向PA0写1

延迟

向PA0写0

5:等待测量结束

CC1标志位从0变到1——捕获到了上升沿

CC2的标志位从0变到1 ——捕获到了下降沿

__HAL_TIM_GET_FLAG(__HANDLE__,__FLAG__)

返回值0——标志位等于0

返回值非零——标志位等于1

6:关闭定时器

7:计算测量结果

uint16_t ccr1 = __HAL_TIM_GET_COMPARE(&htim1 , TIM_CHANNEL_1);  //计算CCR1的值

 uint16_t ccr2 = __HAL_TIM_GET_COMPARE(&htim1 , TIM_CHANNEL_2);  //计算CCR2的值

float pulseWidth = (cc1 - ccr2) *1e-6f;

float distance = 340.0f *pulseWidth / 2.0f;

8:通过判断距离点亮/熄灭板载LED

六、代码

/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file           : main.c
  * @brief          : Main program body
  ******************************************************************************
  * @attention
  *
  * Copyright (c) 2025 STMicroelectronics.
  * All rights reserved.
  *
  * This software is licensed under terms that can be found in the LICENSE file
  * in the root directory of this software component.
  * If no LICENSE file comes with this software, it is provided AS-IS.
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "tim.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */

/* 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_TIM1_Init();
  /* USER CODE BEGIN 2 */

  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
		//1.让计数器CNT归零
		__HAL_TIM_SET_COUNTER(&htim1 , 0);
		
		//2.清除CC1/CC2的标志位
		__HAL_TIM_CLEAR_FLAG(&htim1 , TIM_FLAG_CC1);
		__HAL_TIM_CLEAR_FLAG(&htim1 , TIM_FLAG_CC2);
		
		//3.启动输入捕获
		HAL_TIM_IC_Start(&htim1 , TIM_CHANNEL_1);
		HAL_TIM_IC_Start(&htim1 , TIM_CHANNEL_2);
		
		//4.向Trig发送脉冲
		
		 HAL_GPIO_WritePin(GPIOA , GPIO_PIN_0 , GPIO_PIN_SET);
		
		 for (uint32_t i = 0 ; i < 10 ; i++);
		
		 HAL_GPIO_WritePin(GPIOA , GPIO_PIN_0 , GPIO_PIN_RESET);
		
		//5.等待测量结束
		uint8_t success = 0;
		uint32_t expireTime = HAL_GetTick() + 50 ;  //计算超时时间
		
		while(expireTime > HAL_GetTick())   //判断是否超时
		{
		 uint32_t cc1Flag = __HAL_TIM_GET_FLAG(&htim1,TIM_FLAG_CC1);    //CC1标志位
		 uint32_t cc2Flag = __HAL_TIM_GET_FLAG(&htim1,TIM_FLAG_CC2);    //CC2标志位
			
			if (cc1Flag && cc2Flag )
			{
				success = 1 ;
				break;
			}
	
		}
		
		//6.关闭定时器
		
		HAL_TIM_IC_Stop(&htim1 , TIM_CHANNEL_1);   
		HAL_TIM_IC_Stop(&htim1 , TIM_CHANNEL_1);
		
		//7.计算测量结果
		 
		if(success == 1)
		{
			uint16_t ccr1 = __HAL_TIM_GET_COMPARE(&htim1 , TIM_CHANNEL_1); //计算CCR1的值

      uint16_t ccr2 = __HAL_TIM_GET_COMPARE(&htim1 , TIM_CHANNEL_2);//计算CCR2的值

      float pulseWidth = (ccr1 - ccr2) *1e-6f;    //计算脉宽  秒 = 微秒 * 1e-6

      float distance = 340.0f *pulseWidth / 2.0f;  //计算距离 
			
			if(distance < 0.2)  //距离是否小于0.2米
			{
				HAL_GPIO_WritePin(GPIOC , GPIO_PIN_13 , GPIO_PIN_RESET);   //点亮板载LED
			}
			else 
			{
				HAL_GPIO_WritePin(GPIOC , GPIO_PIN_13 , GPIO_PIN_SET);     //熄灭板载LED
			}
		}
		
		
    /* USER CODE END WHILE */

    /* 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_HSI;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
  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_HSI;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != 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 */

最后将代码下载编译,就可以看到当物体距离超声波传感器0.2m以内时板载LED点亮,大于0.2米时板载LED熄灭。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

LCC并行光模块、PLCC并行光模块和POB并行光模块的区别

一、封装结构与材料差异 PLCC&#xff08;Plastic Leaded Chip Carrier&#xff09; 封装材料&#xff1a;采用塑料材质&#xff0c;引脚数量较多&#xff08;20-84个&#xff09;&#xff0c;引脚分布在四周12散热设计&#xff1a;通常通过PCB基板嵌装铜散热片提升散热性能8…

WIFI的SSID超长,隐藏,重复 (2.4G和5G差异)

目录 1、2.4G和5G的频率范围‌ 2、2.4G和5G的差异‌&#xff1a; 3、隐藏ssid显示为\x00 4、 重复的ssid名称 扩展 前言 最近处理wifi设备时发现&#xff0c;小小一个ssid就有超多的问题。 不是中文转义就是超长&#xff0c;现在还发现空字符的&#xff0c;原来时对方路由隐藏了…

FS800DTU联动OneNET平台数据可视化View

目录 1 前言 2 环境搭建 2.1 硬件准备 2.2 软件环境 2.3 硬件连接 3 注册OneNET云平台并建立物模型 3.1 参数获取 3.2 连接OneNET 3.3上报数据 4 数据可视化View 4.1 用户信息获取 4.2 启用数据可视化View 4.3 创建项目 4.4 编辑项目 4.5 新增数据源 4.6 数据过滤器配置 4.6 项…

Linux时间日期类指令

1、data指令 基本语法&#xff1a; date &#xff1a; 显示当前时间date %Y : 显示当前年份date %m &#xff1a; 显示当前月份date %d &#xff1a; 显示当前哪一天date “%Y-%m-%d %H:%M:%S" &#xff1a; 显示年月日时分秒date -s 字符串时间 &#xff1a; 设置系统时…

Elasticsearch:使用经过训练的 ML 模型理解稀疏向量嵌入

作者&#xff1a;来自 Elastic Dai Sugimori 了解稀疏向量嵌入&#xff0c;理解它们的作用/含义&#xff0c;以及如何使用它们实现语义搜索。 Elasticsearch 提供语义搜索功能&#xff0c;允许用户使用自然语言进行查询并检索相关信息。为此&#xff0c;目标文档和查询必须首先…

MagicArticulate: 超48K海量数据革新3D动画,自回归Transformer驱动关节智能生成!

论文链接&#xff1a;https://arxiv.org/pdf/2502.12135 Git链接&#xff1a;https://chaoyuesong.github.io/MagicArticulate/ 亮点直击 首个大规模关节化基准数据集&#xff0c;包含超过33,000个具有高质量关节标注的模型&#xff1b; 一种新颖的两阶段框架&#xff0c;有效…

【洛谷入门赛】B4019 皆与生物有缘

题意 两个老师会对你进行打分&#xff0c;我们要做的是把两个老师每题给的分全加起来&#xff0c;然后 2 \div 2 2 再向上取整&#xff0c;输出这个值即可。 思路 首先&#xff0c;我们要输入。 根据我们总结的题意&#xff0c;我们还需要把所有分数加起来。 因为我们尽…

如何选择更安全的无人自助管理私人影院物联网框架?

对于无人自助管理的私人电影院来说&#xff0c;安全是至关重要的&#xff0c;在物联网层面上&#xff0c;更要确保其相关安全措施&#xff0c;才能更好地安全运转。选择更安全的无人自助管理私人影院物联网框架时&#xff0c;可以从以下几个关键方面进行评估&#xff1a; 1. 安…

基本网络安全的实现

基本网络安全的实现 一 &#xff1a;AAA AAA 是Authentication&#xff0c;Authorization and Accounting&#xff08;认证、授权和计费&#xff09;的简 称&#xff0c;它提供了一个用来对认证、授权和计费这三种安全功能进行配置的一致性框架&#xff0c; 它是对网络安全…

【大模型】Ubuntu下 fastgpt 的部署和使用

前言 本次安装的版本为 fastgpt:v4.8.8-fix2。 最新版本fastgpt:v4.8.20-fix2 问答时报错&#xff0c;本着跑通先使用起来&#xff0c;就没有死磕下去&#xff0c;后面bug解了再进行记录。   github连接&#xff1a;https://github.com/labring/FastGPT fastgpt 安装说明&…

python--泰坦尼克号人员存活模型训练+预测

1.先到Kaggle平台下载关于泰坦尼克号的数据。 泰坦尼克号数据集&#xff1a;这是Kaggle上的经典数据集&#xff0c;用于预测乘客是否幸存。 地址&#xff1a;Titanic - Machine Learning from Disaster | Kaggle 包含以下文件&#xff1a; train.csv&#xff1a;训练集&#…

【Azure 架构师学习笔记】- Azure Databricks (12) -- Medallion Architecture简介

本文属于【Azure 架构师学习笔记】系列。 本文属于【Azure Databricks】系列。 接上文 【Azure 架构师学习笔记】- Azure Databricks (11) – UC搭建 前言 使用ADB 或者数据湖&#xff0c;基本上绕不开一个架构“Medallion”&#xff0c; 它使得数据管理更为简单有效。ADB 通过…

spring中的注解介绍

本篇文章专门用来介绍spring中的各种注解。 1、RestController 1、含义 2、举例 3、使用场景 RestController 通常用于开发 RESTful API&#xff0c;适合返回 JSON 或 XML 数据的场景 4、总结 RestController 是 Spring 中用于简化 RESTful Web 服务开发的注解&#xff0c;它结…

嵌入式开发:傅里叶变换(5):STM32和Matlab联调验证FFT

目录 1. MATLAB获取 STM32 的原始数据 2. 将数据上传到电脑 3. MATLAB 接收数据并验证 STM32进行傅里叶代码 结果分析 STM32 和 MATLAB 联调是嵌入式开发中常见的工作流程&#xff0c;通常目的是将 STM32 采集的数据或控制信号传输到 MATLAB 中进行实时处理、分析和可视化…

Linux设备驱动开发-SPI驱动开发详解(包含设备树处理详细过程)

基础知识及 SPI 相关结构体介绍 引脚&#xff1a;MISO&#xff08;master 输入&#xff0c;slave 输出&#xff09;&#xff0c;MOSI&#xff08;master 输出&#xff0c;slave 输入&#xff09;&#xff0c;片选引脚&#xff0c;SCK&#xff08;时钟&#xff09; 控制寄存器&…

嵌入式八股文(五)硬件电路篇

一、名词概念 1. 整流和逆变 &#xff08;1&#xff09;整流&#xff1a;整流是将交流电&#xff08;AC&#xff09;转变为直流电&#xff08;DC&#xff09;。常见的整流电路包括单向整流&#xff08;二极管&#xff09;、桥式整流等。 半波整流&#xff1a;只使用交流电的正…

C++初阶——简单实现stack和queue

目录 1、Deque(了解) 1.1 起源 1.2 结构 1.3 优缺点 1.4 应用 2、Stack 3、Queue 4、Priority_Queue 注意&#xff1a;stack&#xff0c;queue&#xff0c;priority_queue是容器适配器(container adaptor) &#xff0c;封装一个容器&#xff0c;按照某种规则使用&#…

【Git】六、企业级开发模型

文章目录 Ⅰ. 前言Ⅱ. 系统开发环境Ⅲ. Git 分支设计规范master分支release分支develop分支feature分支hotfix分支 Ⅰ. 前言 ​ 我们知道&#xff0c;一个软件从零开始到最终交付&#xff0c;大概包括以下几个阶段&#xff1a;规划、编码、构建、测试、发布、部署和维护。 ​…

Apache SeaTunnel 构建实时数据同步管道(最新版)

文章作者 王海林 白鲸开源 数据集成引擎研发 Apache SeaTunnel Committer & PMC Member&#xff0c;Apache SkyWalking Committer&#xff0c;多年平台研发经验&#xff0c;目前专注于数据集成领域。 导读 在当今数字化快速发展的时代&#xff0c;数据已然成为企业决策…

在 Windows 上配置 Ollama 服务并开放局域网访问

为了在局域网内共享 Ollama 服务&#xff0c;我们需要完成以下两步&#xff1a; 1、设置 Ollama 的环境变量 OLLAMA_HOST&#xff0c;使其监听局域网的 IP 地址。 &#xff08;1&#xff09; 配置 Ollama 服务的监听地址 Ollama 服务使用环境变量 OLLAMA_HOST 来指定监听的地…