STM32——玩转超声波传感器

news2024/11/24 9:04:34

目录

1.什么是超声波?

超声波的基本特点:

2.超声波传感器介绍:HC-SR04

HC-SR04 主要特点:

HC-SR04 接线如下:

HC-SR04 工作原理:

如何编写超声波测距代码?

编写逻辑:

编写思路:

1.配置GPIO口:

2.初始化定时器:

3.利用超声波来回时间计算距离:

完整代码:

hcsr04.c

hcsr04.h

main.c


1.什么是超声波?

        声音是由物体振动产生的,人能听到的频率在20Hz~20kHz,超声波是指频率高于人类听觉范围的声波

        频率小于20Hz的叫次声波,频率大于20kHz的叫超声波

超声波的基本特点:


        1.高频率:超声波的频率超过20,000 Hz,一般应用中使用的频率通常在几百千赫兹到几兆赫兹(MHz)之间。
        2. 高指向性:超声波具有较强的方向性,容易集中成一束,传播时能量集中,因此可以被精确地控制和引导。
        3. 穿透性强:超声波可以穿透许多材料,尤其是液体和人体组织,所以广泛用于医学成像。
        4. 反射和散射:超声波遇到不同材料的界面时,会发生反射和散射,这一特性在超声波成像和检测中非常有用。

2.超声波传感器介绍:HC-SR04

        HC-SR04 是一种常用的超声波距离传感器,广泛用于机器人、测距、物体检测等领域。它通过发射超声波并检测回波来测量与物体的距离,工作原理类似于蝙蝠或海豚的回声定位。

HC-SR04 主要特点:

  • 探测距离:2~600cm
  • 探测精度:0.1cm±1%
  • 感应角度:<15°
  • 输出方式:GPIO
  • 工作电压:DC 3~5.5V
  • 工作电流:5.3mA
  • 工作温度:-40~85℃

HC-SR04 接线如下:

HC-SR04STM32备注
VCC3.3/5V外接直流电源
Trig(触发信号)任意一个GPIO口超声波输入端——单片机输出
Echo(回波信号)任意一个GPIO口超声波输出端——单片机输入
GNDGND接地

HC-SR04 工作原理:

1.向Trig引脚发送一个10us的高电平脉冲,传感器开始发射超声波。
2. 超声波遇到障碍物后反射回传感器,Echo 引脚输出一个高电平信号,其持续时间与超声波从发射到返回的时间成正比。
3. 根据声音在空气中的传播速度(约343米/秒),通过公式—距离 = (高电平时间 × 声速) / 2 计算出物体与传感器的距离。

正常时序流程:

  1. 单片机给超声波模块发送大于 10us 的高电平的触发信号;
  2. 超声波模块收到触发信号后 Trig 端发送 8个40kHz 的超声波脉冲;
  3. Echo 端由低电平转为高电平,同时开始发送超声波;
  4. 超声波模块检测到返回信号,Echo 端由高电平转为低电平;
  5. Echo 端高电平宽度即为超声波传播时间。

如何编写超声波测距代码?

编写逻辑:

1.怎么发送超声波?

        单片机给超声波模块Trig引脚发送大于 10us 的高电平的触发信号

2.怎么知道发出来了超声波?

        超声波模块的Echo 端低电平转为高电平,表示开始发送超声波;

3.怎么知道接收到了超声波?

        超声波模块Echo 端高电平转为低电平,表示检测到返回信号;

4.怎么计算时间?

        Echo引脚维持高电平的时间

(在超声波发出的那一刻我们启动定时器,波返回来的那一刻定时器停止计数)

5.怎么计算距离?

        距离 = (高电平时间 × 声速) / 2

编写思路:

1.配置GPIO口:

首先我们应该先看一下用哪个引脚来接入超声波模块——配置GPIO口

        我们选用GPIOB的Pin6和Pin7脚,并将他们在头文件中宏定义,方便之后的调用,使代码更加清晰。(其中Pin6脚的作用是输出大于10us的高电平信号给Trig、Pin7脚的作用是接收Echo回波信号——所以在这里我们并不用特别去选取引脚,只要能正常使用就行,只需注意给他相应的配置。

#define TRIG_PORT                   GPIOB
#define TRIG_PIN                    GPIO_PIN_6

#define ECHO_PORT                   GPIOB
#define ECHO_PIN                    GPIO_PIN_7

 配置GPIO口时,首先要记得开启时钟:

#define TRIG_GPIO_CLK_ENABLE()      __HAL_RCC_GPIOB_CLK_ENABLE()
#define ECHO_GPIO_CLK_ENABLE()      __HAL_RCC_GPIOB_CLK_ENABLE()

 然后调用:HAL_GPIO_Init();——进行初始化

void hcsr04_gpio_init(void)
{
    GPIO_InitTypeDef gpio_initstruct;
    //打开时钟
    TRIG_GPIO_CLK_ENABLE();
    ECHO_GPIO_CLK_ENABLE();
    
    //初始化Trig引脚
    gpio_initstruct.Pin = TRIG_PIN;
    gpio_initstruct.Mode = GPIO_MODE_OUTPUT_PP;     //推挽输出模式        
    gpio_initstruct.Pull = GPIO_NOPULL;   //不上拉也不下拉                  
    gpio_initstruct.Speed = GPIO_SPEED_FREQ_HIGH;        //引脚设置为高速   
    HAL_GPIO_Init(TRIG_PORT, &gpio_initstruct);
    
	//初始化Echo引脚
    gpio_initstruct.Pin = ECHO_PIN;
    gpio_initstruct.Mode = GPIO_MODE_INPUT;       //输入模式,接收Echo的回波信号      
    HAL_GPIO_Init(ECHO_PORT, &gpio_initstruct);
}
2.初始化定时器:

然后,我们需要超声波测距,需要知道超声波的来回时间具体是多少,所以我们需要初始化一个定时器,用于测量 Echo 高电平宽度——这里我们初始化了通用定时器2

我们先调用:HAL_TIM_Base_Init(&tim2_handle);—对定时器2进行初始化

然后调用void HAL_TIM_Base_MspInit(TIM_HandleTypeDef *htim)函数初始化定时器的时钟。

MSP函数主要负责外设的低层初始化(如时钟、引脚等)

TIM_HandleTypeDef tim2_handle = {0};//定时器句柄结构体,存储定时器 TIM2 的配置信息

//定时器初始化函数
void tim2_init(void)
{
    tim2_handle.Instance = TIM2;	//指定定时器为Tim2
    tim2_handle.Init.Prescaler = 72 - 1;	//分频系数
    tim2_handle.Init.Period = 65536 - 1;	//定时器周期—计数器从0记到65535
    tim2_handle.Init.CounterMode = TIM_COUNTERMODE_UP;		//向上计数
    tim2_handle.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;	
	//定时器溢出后,定时器的自动重载值会立即生效
    HAL_TIM_Base_Init(&tim2_handle);
}

/*msp函数——主要负责外设的低层初始化:如时钟、引脚*/
void HAL_TIM_Base_MspInit(TIM_HandleTypeDef *htim)
{
    if(htim->Instance == TIM2)	//该函数检查传递过来的 htim 句柄是否属于TIM2 实例
    {
        __HAL_RCC_TIM2_CLK_ENABLE();	//启用定时器时钟
    }
}

其中,对于 分频系数和定时周期的计算,我们利用定时器溢出时间计算的公式。

  tim2_handle.Init.Prescaler = 72 - 1;	//分频系数
  tim2_handle.Init.Period = 65536 - 1;	//定时器周期—计数器从0记到65535

 (这里的(PSC+1)/Ft代表的意思是计一个数所花的时间)

                72/72M=1us                ARR+1=65536表示一直记到65.536ms计数器才溢出

3.利用超声波来回时间计算距离:

首先,我们需要封装四个函数:

/*启动定时器*/
void tim2_start(void)
{
    HAL_TIM_Base_Start(&tim2_handle);
}
/*停止定时器*/
void tim2_stop(void)
{
    HAL_TIM_Base_Stop(&tim2_handle);
}
/*获取定时器的值*/
uint16_t tim2_get_cnt(void)
{
    return __HAL_TIM_GetCounter(&tim2_handle);//该函数可以返回当前计数器的值
}
/*设置定时器的值*/
void tim2_set_cnt(uint16_t val)
{
    __HAL_TIM_SetCounter(&tim2_handle, val);//该函数可以将计数器值设置为传入参数val
}

 然后我们利用之前HC-SR04的工作原理列出的逻辑进行编写代码

float hcsr04_get_length(void)
{
	uint16_t total_time=0;
	float distance = 0;
	
	/*首先发出超过10us的高电平信号,超声波模块发出超声波*/
	TRIG_HIGH();
	delay_us(15);
	TRIG_LOW();
	
	while(ECHO_STATUS()==RESET);//低电平的时候就一直不读取
	/*Echo引脚,由低电平跳转到高电平,表示开始发送波-计时开始*/
	tim2_start();
	tim2_set_cnt(0);
	
	while(ECHO_STATUS()==SET);//高电平的时候就一直计数
	/*Echo,由高电平跳转回低电平,表示波回来了-计数停止*/
	tim2_stop();
	/*计算超声波来回时长*/
	total_time = tim2_get_cnt();
	
	/*计算距离- 距离 = 速度(343m/s) * 时间*/
	distance = total_time * 0.01715;
	
	return distance;
}

完整代码:

hcsr04.c
#include "hcsr04.h"
#include "delay.h"

TIM_HandleTypeDef tim2_handle = {0};//定时器句柄结构体,存储定时器 TIM2 的配置信息

//定时器初始化函数
void tim2_init(void)
{
    tim2_handle.Instance = TIM2;	//指定定时器为Tim2
    tim2_handle.Init.Prescaler = 72 - 1;	//分频系数
    tim2_handle.Init.Period = 65536 - 1;	//定时器周期—计数器从0记到65535
    tim2_handle.Init.CounterMode = TIM_COUNTERMODE_UP;		//向上计数
    tim2_handle.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;	
	//定时器溢出后,定时器的自动重载值会立即生效
    HAL_TIM_Base_Init(&tim2_handle);
}

/*msp函数——主要负责外设的低层初始化:如时钟、引脚*/
void HAL_TIM_Base_MspInit(TIM_HandleTypeDef *htim)
{
    if(htim->Instance == TIM2)	//该函数检查传递过来的 htim 句柄是否属于TIM2 实例
    {
        __HAL_RCC_TIM2_CLK_ENABLE();	//启用定时器时钟
    }
}

/*启动定时器*/
void tim2_start(void)
{
    HAL_TIM_Base_Start(&tim2_handle);
}
/*停止定时器*/
void tim2_stop(void)
{
    HAL_TIM_Base_Stop(&tim2_handle);
}
/*获取定时器的值*/
uint16_t tim2_get_cnt(void)
{
    return __HAL_TIM_GetCounter(&tim2_handle);//该函数可以返回当前计数器的值
}
/*设置定时器的值*/
void tim2_set_cnt(uint16_t val)
{
    __HAL_TIM_SetCounter(&tim2_handle, val);//该函数可以将计数器值设置为传入参数val
}

void hcsr04_gpio_init(void)
{
    GPIO_InitTypeDef gpio_initstruct;
    //打开时钟
    TRIG_GPIO_CLK_ENABLE();
    ECHO_GPIO_CLK_ENABLE();
    
    //初始化Trig引脚
    gpio_initstruct.Pin = TRIG_PIN;
    gpio_initstruct.Mode = GPIO_MODE_OUTPUT_PP;     //推挽输出模式        
    gpio_initstruct.Pull = GPIO_NOPULL;   //不上拉也不下拉                  
    gpio_initstruct.Speed = GPIO_SPEED_FREQ_HIGH;        //引脚设置为高速   
    HAL_GPIO_Init(TRIG_PORT, &gpio_initstruct);
    
	//初始化Echo引脚
    gpio_initstruct.Pin = ECHO_PIN;
    gpio_initstruct.Mode = GPIO_MODE_INPUT;             
    HAL_GPIO_Init(ECHO_PORT, &gpio_initstruct);
}

void hcsr04_init(void)
{
    tim2_init();
    hcsr04_gpio_init();
}

float hcsr04_get_length(void)
{
	uint16_t total_time=0;
	float distance = 0;
	
	/*首先发出超过10us的高电平信号,超声波模块发出超声波*/
	TRIG_HIGH();
	delay_us(15);
	TRIG_LOW();
	
	while(ECHO_STATUS()==RESET);//低电平的时候就一直不读取
	/*高电平来了-计时开始*/
	tim2_start();
	tim2_set_cnt(0);
	
	while(ECHO_STATUS()==SET);//高电平的时候就一直计数
	/*低电平来了-计数停止*/
	tim2_stop();
	/*计算超声波来回时长*/
	total_time = tim2_get_cnt();
	
	/*计算距离*/
	distance = total_time * 0.01715;
	
	return distance;
}
	
hcsr04.h
#ifndef __HCSR04_H__
#define __HCSR04_H__

#include "sys.h"

#define TRIG_PORT                   GPIOB
#define TRIG_PIN                    GPIO_PIN_6
#define TRIG_GPIO_CLK_ENABLE()      __HAL_RCC_GPIOB_CLK_ENABLE()
#define TRIG_HIGH()                 HAL_GPIO_WritePin(TRIG_PORT, TRIG_PIN, GPIO_PIN_SET)
#define TRIG_LOW()                  HAL_GPIO_WritePin(TRIG_PORT, TRIG_PIN, GPIO_PIN_RESET)

#define ECHO_PORT                   GPIOB
#define ECHO_PIN                    GPIO_PIN_7
#define ECHO_GPIO_CLK_ENABLE()      __HAL_RCC_GPIOB_CLK_ENABLE()
#define ECHO_STATUS()               HAL_GPIO_ReadPin(ECHO_PORT, ECHO_PIN)

void hcsr04_init(void);
float hcsr04_get_length(void);

#endif
main.c
#include "sys.h"
#include "delay.h"
#include "led.h"
#include "uart1.h"
#include "hcsr04.h"

int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9);     /* 设置时钟, 72Mhz */
    uart1_init(115200);
    hcsr04_init();

    while(1)
    { 
        printf("dis: %.2f\r\n", hcsr04_get_length());
        delay_ms(1000);
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【2024】前端学习笔记4-图像标记

在 HTML 中&#xff0c;<img>标签用于在网页中插入图像。 基本语法&#xff1a; <img src"图像文件地址" alt"替代文本">src&#xff1a;指定图片的路径&#xff0c;可以是相对路径、绝对路径。alt&#xff1a;为图像提供代替文本&#xff0…

9.12日常记录

1.extern关键字 1&#xff09;诞生动机:在一个C语言项目中&#xff0c;需要再多个文件中使用同一全局变量或是函数&#xff0c;那么就需要在这些文件中再声明一遍 2&#xff09;用于声明在其他地方定义的一个变量或是函数&#xff0c;在当前位置只是声明&#xff0c;告诉编译器…

【办公类】幼儿健康数据模版批量更改日期(保健老师填写)

背景需求 今天下发通知 三个园区的保健老师需要填写 1.2023学年&#xff08;202406&#xff09;的六一体检数据 2.2024学年&#xff08;202409&#xff09;的新生入园体检数据 我先把上一轮填写过的数据模版下载下来&#xff08;套用模版&#xff09; 把EXCEL下载到原始文件…

驾驭不断发展的人工智能世界

从很多方面来看&#xff0c;历史似乎正在重演。许多企业正争相采用生成式人工智能 (Gen AI)&#xff0c;就像它们争相采用云计算一样&#xff0c;原因也是一样的&#xff1a;效率、成本节约和竞争优势。 然而&#xff0c;与云一样&#xff0c;GenAI 仍是一项发展中的技术&…

机器学习(西瓜书)第 9 章 聚类

9.1 聚类任务和距离计算 在”无监督学习“中&#xff0c;训练样本的标记信息是未知的&#xff0c;目标是通过对无标记训练样本的学习来揭示数据的内在性质及规律&#xff0c;为进一步的数据分析提供基础.此类学习任务中研究最多、应用最广的是“聚类”(clustering). 聚类试图…

咖啡果实病虫害检测系统源码分享

咖啡果实病虫害检测检测系统源码分享 [一条龙教学YOLOV8标注好的数据集一键训练_70全套改进创新点发刊_Web前端展示] 1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 项目来源AACV Association for the Advancement of Comput…

【最新综述】基于深度学习的超声自动无损检测(上)

Deep learning in automated ultrasonic NDE – Developments, axioms and opportunities 传统上&#xff0c;超声无损检测数据分析一直由训练有素的操作员在基本自动化工具的支持下手动解释数据。最近&#xff0c;开始出现许多针对个别无损检测任务&#xff08;数据预处理、缺…

即插即用篇 | YOLOv10 引入矩形自校准模块RCM | ECCV 2024

本改进已同步到YOLO-Magic框架! 语义分割是许多应用的重要任务,但要在有限的计算成本下实现先进性能仍然非常具有挑战性。在本文中,我们提出了CGRSeg,一个基于上下文引导的空间特征重建的高效且具有竞争力的分割框架。我们精心设计了一个矩形自校准模块,用于空间特征重建和…

HarmonyOS开发实战( Beta5.0)橡皮擦案例实践详解

鸿蒙HarmonyOS开发往期必看&#xff1a; HarmonyOS NEXT应用开发性能实践总结 最新版&#xff01;“非常详细的” 鸿蒙HarmonyOS Next应用开发学习路线&#xff01;&#xff08;从零基础入门到精通&#xff09; 介绍 本示例通过ohos.graphics.drawing库和blendMode颜色混合实…

Obsidian git sync error / Obsidian git 同步失敗

Issue: commit due to empty commit message Solution 添加commit資訊&#xff0c;確保不留空白 我的設置&#xff1a;auto-backup: {{hostname}}/{{date}}/

ESXI8.0 vsphere vcenter 多网卡多网段配置

一般来说服务器至少两块网卡&#xff0c;安装esxi后一种方案是利用闲置网卡建立多上传链路&#xff0c;聚合&#xff0c;另一种是配置多网段进行虚拟机隔离&#xff0c;网上也没找到讲的很清楚的&#xff0c;经过多种尝试终于学会&#xff0c;记录分享一下 首先物理交换机的随…

谷歌开发者账号被封会影响AdMob吗?收款卡更换修改有什么限制?

众所周知&#xff0c;AdMob作为谷歌旗下的广告平台&#xff0c;是众多开发者的主要变现渠道之一。那在开发者账号被封的情况下&#xff0c;AdMob账号是否会也受到波及&#xff1f; 虽说通常谷歌开发者账号被封并不直接等同于AdMob账号被封&#xff0c;但还是会互相影响的。即使…

C 和 C++ struct 的区别?

目录 1. 默认访问权限 2. 面向对象特性 3. 继承和多态 4. 访问控制 5. 使用习惯 6. 默认继承权限 1. 默认访问权限 C 语言&#xff1a;在 C 中&#xff0c;struct 中的所有成员默认是 public&#xff08;可直接访问&#xff09;。C 语言&#xff1a;在 C 中&#xff0c;…

BFS广度优先搜索解决迷宫问题

前言 BFS广度优先搜索解决迷宫问题 迷宫问题 原题目&#xff1a;迷宫由n行m列的单元格组成(n,m都小于等于50)&#xff0c;每个单元格要吗是空地要吗是障碍物。现在请你找到一条从起点到终点的最短路径长度。 分析 首先我们将起点入队&#xff0c; 然后队首节点可拓展的点入…

总线性能指标及标准

一、总线性能指标 1.总线周期 一次总线操作所需的时间&#xff08;包括申请阶段、寻址阶段、传输阶段和结束阶段&#xff09;&#xff0c;通常 由若干个总线时钟周期构成。 2.总线时钟周期 即机器的时钟周期。计算机有一个统一的时钟&#xff0c;以控制整个计算机的各个部件…

Android 系统下:普通应用无缝安装,Launcher 应用安装遭遇罕见障碍解析

目录 一、场景 1.1 错误分析 1.2 解决方法尝试 1.2.1 检查应用的安装位置 1.2.2 使用ADB安装 1.2.3 检查APK的签名 1.2.4 检查可用的内部存储空间 1.2.5 将应用程序安装到设备的内部存储空间 1.2.6 重置设备&#xff08;谨慎使用&#xff09; 1.2.7 获取Root权限&…

【Qt】选择器

选择器概况 QSS选择器支持以下几种&#xff1a; 选择器示例说明 全局选择器 * 选择所有的 widget. 类型选择器 (type selector) QPushButton 选择所有的 QPushButton 和 其⼦类 的控件. 类选择器 (class selector) .QPushButton 选择所有的 QPushButton 的控件. 不会选…

C++——STL(list类)

1.list的介绍 1. list是可以在常数范围内在任意位置进行插入和删除的序列式容器&#xff0c;并且该容器可以前后双向迭代。 2. list的底层是双向链表结构&#xff0c;双向链表中每个元素存储在互不相关的独立节点中&#xff0c;在节点中通过指针指向其前一个元素和后一个元素…

一文400字看懂,如何评估系统负载指标load average

在linux系统输入指令top后可以看到系统近1分钟、近5分钟、近15分钟的平均负载。 但是很多人工作10年也未必知道这个指标的含义&#xff0c;到底多少表示系统已经忙不过来&#xff1f; 如图&#xff0c;最近1分钟、5分钟&#xff0c;15分钟系统平均负载分别是0.10&#xff0c;0…

什么是充放电测试?怎么测试电源的充放电性能?

对于需要不间断电源供应的电子系统而言&#xff0c;某些电源模块通过搭载电池来应对突发停电或其他系统故障。针对这些模块的电池性能保障&#xff0c;充放电测试显得尤为重要&#xff0c;它是确保电源稳定性的关键测试方法。 充放电测试原理 充放电测试是指通过模拟电池在实际…