STM32——玩转超声波传感器

目录

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;	//指定定时器为Tim2tim2_handle.Init.Prescaler = 72 - 1;	//分频系数tim2_handle.Init.Period = 65536 - 1;	//定时器周期—计数器从0记到65535tim2_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;	//指定定时器为Tim2tim2_handle.Init.Prescaler = 72 - 1;	//分频系数tim2_handle.Init.Period = 65536 - 1;	//定时器周期—计数器从0记到65535tim2_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.rhkb.cn/news/422724.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JavaWeb项目打包、部署至Tomcat并启动的全程指南(图文详解)

前言 我们想要部署一个javaWeb项目到tomcat上&#xff0c;需要了解一些概念 什么是tomcat&#xff1f; Tomcat 是 Apache 软件基金会&#xff08;Apache Software Foundation&#xff09;下的一个开源项目&#xff0c;主要用于实现 Java Servlet、JavaServer Pages&#xff08;…

SpinalHDL之数据类型(六)

本文作为SpinalHDL学习笔记第五十九篇,介绍SpinalHDL的Vec数据类型。 目录: 1.描述(Description) 2.声明(Declaration) 3.操作符(Operators) ⼀、描述(Description) Vec是定义了⼀组带有标号的信号的复合信号(基于SpinalHDL基础类别)。 ⼆、声明(Declaration) 声明向量的…

远程桌面内网穿透是什么?有什么作用?

远程桌面内网穿透指的是通过特定技术手段&#xff0c;将处于内网中的电脑或服务器&#xff0c;通过外部网络&#xff08;互联网&#xff09;进行访问。内网穿透的主要作用是解决在内网环境下&#xff0c;远程设备与外部互联网之间的连接问题&#xff0c;允许用户从外部访问内网…

Docker部署tenine实现后端应用的高可用与负载均衡

采用Docker方式的Tengine 和 keepalived 组合模式可以实现小应用场景的高可用负载均衡需求 目录 网络架构一、环境准备二、软件安装1. 下载Tenine镜像2. 下载Keepalived镜像3. 制作SpringBoot镜像 三、软件配置1. 创建应用容器2. 代理访问应用3. 创建Keepalived4. 测试高可用 网…

【计算机网络】UDP 协议详解及其网络编程应用

文章目录 一、引言二、UDP1、UDP的协议格式2、UDP 报文的解包和分用3、UDP面向数据报的特点 三、UDP输入输出四、UDP网络编程 一、引言 UDP&#xff08;User Datagram Protocol&#xff0c;用户数据报协议&#xff09;是一种网络通信协议&#xff0c;它属于传输层的协议。是一…

【Qt】子控件选择器

子控件选择器 有些控件内部包含了多个 "⼦控件" . ⽐如 QComboBox 的下拉后的⾯板, ⽐如 QSpinBox 的上下按钮等。 可以通过⼦控件选择器 :: , 针对上述⼦控件进⾏样式设置. 参考⽂档 Qt Style Sheets Reference 中 List of Sub-Controls 例子&#xff1a;设置下拉…

智慧交通:关键技术及应用场景

智慧交通是指通过信息和通信技术&#xff0c;对交通系统进行全面感知、高效管理和智能控制的一种交通管理模式。随着城市化进程的加速和交通需求的增长&#xff0c;智慧交通技术应运而生&#xff0c;为实现交通安全、高效、环保等目标提供了新的途径。 1. 关键技术 物联网技术…

Leetcode Hot 100刷题记录 -Day15(螺旋矩阵)

螺旋矩阵 问题描述&#xff1a; 给你一个 m 行 n 列的矩阵 matrix &#xff0c;请按照 顺时针螺旋顺序 &#xff0c;返回矩阵中的所有元素。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,2,3],[4,5,6],[7,8,9]]输出&#xff1a;[1,2,3,6,9,8,7,4,5] 示例 2&#xff1a; 输…

Docker数据挂载本地目录

docker内的数据映射可以不通过数据卷&#xff0c;直接映射到本地的目录。下面将以mysql容器示例&#xff0c;完成容器的数据映射。 注意&#xff1a;每一个不同的镜像&#xff0c;将来创建容器后内部有哪些目录可以挂载&#xff0c;可以参考DockerHubDocker Hub Container Ima…

07 vue3之组件及生命周期

组件基础 每一个.vue 文件呢都可以充当组件来使用 每一个组件都可以复用 组件的生命周期 简单来说就是一个组件从创建 到 销毁的 过程 成为生命周期 在我们使用Vue3 组合式API 是没有 beforeCreate 和 created 这两个生命周期的 onBeforeMount() 在组件DOM实际渲染安装之前…

跑DecoupleSegNets遇到的问题

论文&#xff1a;[ECCV-2020]: Improving Semantic Segmentation via Decoupled Body and Edge Supervision 代码&#xff1a;https://github.com/lxtGH/DecoupleSegNets 不想用 ./scripts/train/train_cityscapes_ResNet50_deeplab.sh&#xff0c;直接改train.py 原来citisca…

常量指针、指针常量及常量指针常量

目录 1. 常量指针&#xff08;Pointer to Constant&#xff09; 2. 指针常量&#xff08;Constant Pointer&#xff09; 3. 常量指针常量&#xff08;Constant Pointer to Constant&#xff09; 常量指针&#xff1a;指针指向的值不能改&#xff0c;但指针本身可以指向其他地…

Java+selenium+chrome+linux/windows实现数据获取

背景&#xff1a;在进行业务数据获取或者自动化测试时&#xff0c;通常会使用模拟chrome方式启动页面&#xff0c;然后获取页面的数据。在本地可以使用windows的chromedriver.exe进行打开chrome页面、点击等操作。在linux 下通常使用无界面无弹窗的方式进行操作。接下来是实现方…

IP网络广播服务平台任意文件上传漏洞

文章目录 免责声明搜索语法漏洞描述漏洞复现修复建议 免责声明 本文章仅供学习与交流&#xff0c;请勿用于非法用途&#xff0c;均由使用者本人负责&#xff0c;文章作者不为此承担任何责任 搜索语法 icon_hash"-568806419"漏洞描述 该系统在upload接口处可上传任…

YOLOv8改进 | 模块缝合 | C2f 融合RFAConv和CBAM注意力机制 【二次融合 小白必备】

秋招面试专栏推荐 &#xff1a;深度学习算法工程师面试问题总结【百面算法工程师】——点击即可跳转 &#x1f4a1;&#x1f4a1;&#x1f4a1;本专栏所有程序均经过测试&#xff0c;可成功执行&#x1f4a1;&#x1f4a1;&#x1f4a1; 专栏目录 &#xff1a;《YOLOv8改进有效…

自动化任务的错误处理:编写健壮的自动化脚本,处理Office应用中的错误和异常情况

目录 引言 一、自动化任务概述 二、自动化脚本编写基础 2.1 环境准备 2.2 脚本结构 2.3 示例代码 三、Office应用中的错误和异常情况处理 3.1 文件访问权限问题 3.2 文件格式不兼容 3.3 宏病毒和安全性问题 3.4 控件错误和插件问题 四、异常处理与日志记录 4.1 捕…

FPGA技术赋能云数据中心:提高性能与效率

随着现代科技的迅猛发展和大数据时代的推动&#xff0c;云数据中心已成为众多企业的核心基础设施。然而&#xff0c;伴随数据处理需求的不断增长&#xff0c;传统硬件架构在性能、功耗和灵活性方面面临诸多挑战。为了解决这些问题&#xff0c;FPGA&#xff08;现场可编程门阵列…

通信工程学习:什么是MRF多媒体资源功能、MRFC多媒体资源功能控制、MRFP多媒体资源功能处理

一、MRF多媒体资源功能 MRF&#xff08;Multimedia Resource Function&#xff0c;多媒体资源功能&#xff09;是3G/IMS网络中定义的提供多媒体资源功能的网络实体&#xff0c;它为3G/IMS网络的业务和承载提供媒体能力支持。MRF通过提供丰富的媒体处理功能&#xff0c;如播放声…

严重干扰的验证码识别系统源码分享

严重干扰的验证码识别检测系统源码分享 [一条龙教学YOLOV8标注好的数据集一键训练_70全套改进创新点发刊_Web前端展示] 1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 项目来源AACV Association for the Advancement of Comp…

桥接网络设置多用户lxd容器

文章目录 配置宿主机网络固定内核版本安装 lxd、zfs 及 bridge-utils安装宿主机显卡驱动lxd 初始化创建容器模板安装容器显卡驱动复制容器 配置宿主机网络 进入 /etc/netplan/ 目录有一个 yaml 配置文件&#xff0c;下面的命令需要根据自己的 yaml 文件名称自行修改&#xff1…