STM32 超声波模块测距

2023-10-29

HC-SR04模块

在这里插入图片描述

HC-SR04超声波测距模块工作原理

(1)采用IO口TRIG触发测距,给至少10us的高电平信号;

(2)模块自动发送8个40khz的方波,自动检测是否有信号返回;

(3)有信号返回,通过IO口ECHO输出一个高电平,高电平持续的时间就是超声波从发射到返回的时间。测试距离=(高电平时间*声速(340M/S))/2;

(4)本模块使用方法简单,一个控制口发一个10US以上的高电平,就可以在接收口等待高电平输出。一有输出就可以开定时器计时,当此口变为低电平时就可以读定时器的值,此时就为此次测距的时间,方可算出距离。如此不断的周期测,即可以达到你移动测量的值;

HC-SR04超声波模块的时序触发图

在这里插入图片描述

思路步骤

操作思路:
 初始化时将trig和echo端口都置低,首先向给trig 发送至少10 us的高电平脉冲(模块自动向外发送8个40K的方波),然后等待,捕捉 echo 端输出上升沿,捕捉到上升沿的同时,打开定时器开始计时,再次等待捕捉echo的下降沿,当捕捉到下降沿,读出计时器的时间,这就是超声波在空气中运行的时间,按照 测试距离=(高电平时间*声速(340M/S))/2 就可以算出超声波到障碍物的距离。
 
编程步骤

1.配置好相应GPIO,Trig和Echo引脚
2.配置定时器,开启中断,并记录中断产生次数
3.给模块TRIG端口发送大于10us的高电平信号,发出回响信号时,Echo端呈现高电平,此时打开定时器计时;当收到回响信号时,Echo端呈现低电平,此时关闭定时器。
4.获取Echo高电平时间,利用相关公式计算出距离 dis = (高电平时间*声速(340M/S))/2(取平均值获取更精准数据)

代码

ultrasonsic.c

#include "ultrasonsic.h"
#include "delay.h"

u8 msHcCount = 0;//ms计数

static void NVIC_Config()
{
	NVIC_InitTypeDef NVIC_InitStruct;
	//设置中断组为2
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	//设置中断来源
	NVIC_InitStruct.NVIC_IRQChannel										= BASIC_TIM_IRQ;
	NVIC_InitStruct.NVIC_IRQChannelCmd								= ENABLE;
	//设置主优先级
	NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority	= 1;
	//设置次优先级
	NVIC_InitStruct.NVIC_IRQChannelSubPriority				= 1;
	//初始化
	NVIC_Init(&NVIC_InitStruct);
	
	
	
}

void Hcsr04Init()
{  
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;     //生成用于定时器设置的结构体
    GPIO_InitTypeDef GPIO_InitStructure;
	/*开启GPIOB时钟*/
    RCC_APB2PeriphClockCmd(RCC_GPIOx,ENABLE);
     
        //IO初始化
    GPIO_InitStructure.GPIO_Pin =GPIOB_ULTR_Tx_PIN;   //发送电平引脚TX
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//设置推挽输出
    GPIO_Init(GPIOB_ULTR_PORT, &GPIO_InitStructure);
    GPIO_ResetBits(GPIOB_ULTR_PORT,GPIOB_ULTR_Tx_PIN);//一开始低电平
     
    GPIO_InitStructure.GPIO_Pin =   GPIOB_ULTR_Rx_PIN;     //返回电平引脚
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
    GPIO_Init(GPIOB_ULTR_PORT, &GPIO_InitStructure);  
    GPIO_ResetBits(GPIOB_ULTR_PORT,GPIOB_ULTR_Rx_PIN); //默认低电平   
     
     //定时器初始化 使用基本定时器TIM2
    RCC_APB1PeriphClockCmd(BASIC_TIM_CLK, ENABLE);   //使能对应RCC时钟
      //配置定时器基础结构体
    TIM_DeInit(BASIC_TIM);
     //自动重装载值寄存器的值,累计TIM_Period+1个频率后产生一个更新或者中断
		TIM_TimeBaseStructure.TIM_Period			= BASIC_TIM_Period;//设置周期为1000us
		//时钟预分频数
		TIM_TimeBaseStructure.TIM_Prescaler		= BASIC_TIM_Prescaler;//分频数72
	
		//时钟分频因子
		TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1;
		//计数器计数模式,设置向上计数,
		TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up;
		//重复计数器的值
		//	TIM_TimeBaseStructure.TIM_RepetitionCounter=0 ;
    TIM_TimeBaseInit(BASIC_TIM, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位         
        
   // TIM_ClearFlag(BASIC_TIM, TIM_FLAG_Update);   //清除更新中断,免得一打开中断立即产生中断
		TIM_ClearITPendingBit(BASIC_TIM, TIM_FLAG_Update); //清除更新中断,免得一打开中断立即产生中断
    TIM_ITConfig(BASIC_TIM,TIM_IT_Update,ENABLE);    //打开定时器更新中断
		NVIC_Config();
    TIM_Cmd(BASIC_TIM,DISABLE);     
}

void initHcsr04()
{
	Hcsr04Init();
	//BASIC_TIM_NVIC_Config();
}


static void OpenTimer()        //打开定时器
{
        //	/*清除计数器*/
	TIM_SetCounter(BASIC_TIM,0);
	msHcCount = 0;
	TIM_Cmd(BASIC_TIM,ENABLE);//使能定时器
}
 
static void CloseTimer()        //关闭定时器
{
       //	/*关闭计数器使能*/
	TIM_Cmd(BASIC_TIM,DISABLE);
}
 

//定时器2中断服务程序
void TIM2_IRQHandler(void)   //TIM3中断
{
        //	/*判断中断手否真的产生*/
	if(TIM_GetITStatus(BASIC_TIM,TIM_IT_Update) != RESET){
		
		/*清除更新中断标志位*/
		TIM_ClearITPendingBit(BASIC_TIM,TIM_IT_Update);
		msHcCount++;
	}
	
}

//获取定时器时间
u32 GetEchoTimer(void)
{
   u32 time = 0;
	/*//当回响信号很长是,计数值溢出后重复计数,overCount用中断来保存溢出次数*/
	time = msHcCount*1000;//overCount每++一次,代表overCount毫秒,time微妙
	time += TIM_GetCounter(BASIC_TIM);//获取计TIM2数寄存器中的计数值,一边计算回响信号时间
	TIM6->CNT = 0;  //将TIM2计数寄存器的计数值清零
	delay_ms(50);
	return time;
 
}
float Hcsr04GetLength(void )
{
	/*测5次数据计算一次平均值*/
	float length = 0;
	float t = 0;
	float sum = 0;
	u16  	i = 0;
	while(i != 5){
		GPIO_SetBits(GPIOB_ULTR_PORT,GPIOB_ULTR_Tx_PIN);//trig拉高信号,发出高电平
		delay_us(20);//持续时间超过10us
		GPIO_ResetBits(GPIOB_ULTR_PORT,GPIOB_ULTR_Tx_PIN);
		/*Echo发出信号 等待回响信号*/
		/*输入方波后,模块会自动发射8个40KHz的声波,与此同时回波引脚(echo)端的电平会由0变为1;
		(此时应该启动定时器计时);当超声波返回被模块接收到时,回波引 脚端的电平会由1变为0;
		(此时应该停止定时器计数),定时器记下的这个时间即为
			超声波由发射到返回的总时长;*/
		while(GPIO_ReadInputDataBit(GPIOB_ULTR_PORT,GPIOB_ULTR_Rx_PIN) == 0);//echo等待回响
		/*开启定时器*/
		OpenTimer();
		i = i+1; //每收到一次回响信号+1,收到5次就计算均值
		while(GPIO_ReadInputDataBit(GPIOB_ULTR_PORT,GPIOB_ULTR_Rx_PIN) == 1);
		/*关闭定时器*/
		CloseTimer();
		/*获取Echo高电平时间时间*/
		t = GetEchoTimer();
		length = (float)t/58;//单位时cm
		sum += length;		
	}
	length = sum/5;//五次平均值
	
	return length;
}

mian.c

#include "led.h"
#include "delay.h"
#include "ultrasonsic.h"
#include "sys.h"
#include "lcd.h"
#include "usart.h"
#include "beep.h"

 int main(void)
 {	 
			    
	float length;    	   

	delay_init();	    	
	uart_init(115200); //初始化串口,波特率115200 	
	LED_Init();	//初始化LED		
	BEEP_Init();//初始化蜂鸣器 
	initHcsr04();	  							   
	LED0 = 1;//上电默认绿灯亮
		
	while(1){
			
		length=Hcsr04GetLength();	//获取距离
		printf("dis = %fcm\r\n",length);//串口打印距离
		delay_ms(50);
		if(length < 10){//小于10cm亮红灯绿灯灭
			LED0 = 0;
			LED1 = 1;
			BEEP = 1;	
			delay_ms(300);
				
		}else{
			LED0 = 1;
			LED1 = 0;
			BEEP = 0;
		}
			
	}



	} 

实验结果

正常距离 亮绿灯
在这里插入图片描述
有障碍 亮红灯 蜂鸣器报警
在这里插入图片描述
串口显示距离
在这里插入图片描述

总结

期间还是遇到不少问题,坚持冲冲冲。后面加入舵机争取把感应垃圾桶做出来!!!

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

STM32 超声波模块测距 的相关文章

  • RabbitMQ(八)【高级 - 过期时间 TTL】

    八 RabbitMQ高级 过期时间 TTL 上一篇文章 SpringBoot案例 概述 过期时间 TTL 表示可以对消息设置预期的时间 在这个时间内都可以被消费者接收获取 过了之后消息将自动被删除 RabbitMQ可以对 消息和队列 设置
  • 技术博客?

    我认为技术博客是一个非常好的工具 能够帮助我们更好地学习和理解各种技术 同时还能够帮助我们建立自己的品牌和声誉 在我的博客中 我会分享各种最新的技术 开发工具 以及一些我在实际项目中的经验和教训 我的博客主题主要分为以下几个方面 1 前端开

随机推荐

  • 如何高效解决问题?

    前言 我相信很多人在生活中和工作中都会遇到大大小小的问题 每个人解决问题的效率和思维都是有差别的 那么怎么样才能高效解决问题呢 其实问题的产生 从现象到逻辑 考验着我们的知识储备 磨砺着能力的段位 训练着思维的灵活度 从现象到方法论 5W1
  • Elasticsearch学习系列之term和match查询 Elasticsearch查询模式

    Elasticsearch查询模式 一种是像传递URL参数一样去传递查询语句 被称为简单查询 GET library books search 查询index为library type为books的全部内容 GET library book
  • Bean的自动装配 Spring使用注解开发

    Bean的自动装配 自动装配是Spring满足bean依赖的一种方式 Spring会在上下文自动寻找 并自动给bean装配 1 在xml中显示的配置 2 在java中显示的配置 3 隐式 的自动装配bean 重要 测试 autowire b
  • GPT-3 模型

    GPT 3 Generative Pre training Transformer 3 是由 OpenAI 开发的一种大型语言生成模型 它可以用来进行文本生成 翻译 问答等任务 GPT 3 使用了 Transformer 架构 并在大量的网
  • day14-多线程01

    1 实现多线程 1 1简单了解多线程 理解 是指从软件或者硬件上实现多个线程并发执行的技术 具有多线程能力的计算机因有硬件支持而能够在同一时间执行多个线程 提升性能 1 2并发和并行 理解 并行 在同一时刻 有多个指令在多个CPU上同时执行
  • [4G/5G/6G专题基础-160]: BLER与MCS的关系、MCS表格的选择

    目录 第1章 什么是MCS 1 1 基本概念 1 2 三张不同的MCS表 1 3 MCS表的选择 1 4 MCS index的选择 第2章 什么是BLER 2 1 什么是比特误码率BER 2 2 什么是BLER 第3章 BLER与MCS的关
  • 机器学习——建筑能源得分预测

    前言 编码之前是了解我们试图解决的问题和可用的数据 在这个项目中 我们将使用公共可用的纽约市的建筑能源数据 目标是使用能源数据建立一个模型 来预测建筑物的Enerqy Star Score 能源之星分数 并解释结果以找出影响评分的因素 数据
  • 运维必学

    欢迎关注 全栈工程师修炼指南 专注 企业运维实践 网络安全 系统运维 应用开发 物联网实战 全栈文章 等知识分享 设为 星标 每天带你 基础入门 到 进阶实践 再到 放弃学习 作者主页 https www weiyigeek top 博客
  • 猫狗大战 python_猫狗大战

    这个项目我们要做一个识别猫狗的模型 这和上次的数字识别一样 也是运用深度学习 不过这次模型较为复杂 我们会用到迁移学习 站在巨人的肩膀上 借用大佬们已经训练好的模型来搭建我们自己的模型并让它做我们想做的事 安装要求Python3 Numpy
  • python深浅拷贝

    转自 https zhuanlan zhihu com p 258097244 只为记录 侵删 在讲深浅拷贝之前 想先讲一下 is 和 的区别 在进行对象是否相等比较的时候我们可以用 is 和 is 比较两个对象的引用是否相同 即 它们的i
  • 微信小程序模拟车位选择功能(简陋版本)

    功能 实现在线选车位功能 效果图 js文件 const app getApp Page data indicatorDots false autoplay false interval 5000 duration 1000 items va
  • SpringBoot笔记:Log配置和性能比较小结

    文章目录 1 Log的作用 2 Log4j 2 1 简介 2 2 Loggers 2 3 Appenders 2 4 Layouts 2 5 Log4j配置文件示例 3 Log4j2 3 1 Log4j2配置文件示例 4 Logback 4
  • 【终结扩散模型】Consistency Models.OpenAI开源新模型代码,一步成图,1秒18张

    终结扩散模型 Consistency Models OpenAI开源新模型代码 一步成图 1秒18张 0 前言 Abstract 1 Introduction 2 Diffusion Models 3 Consistency Models
  • Java异步I/O编程实现的两种方式:将来式和回调式

    package org zwc test import java io IOException import java nio ByteBuffer import java nio channels AsynchronousFileChan
  • Unity3D Texture2D转换成Sprite格式

    Sprite sprite Sprite Create texture2d new Rect 0 0 64 64 Vector2 zero
  • 一些有用的Java学习资料

    1 资料下载地址库 http zldown diandian com 2 发布一个Java写的俄罗斯方块源码 算法简单 300行 注释详细释详细 http topic csdn net u 20100612 03 a8d7b257 4385
  • service调用规范

    每个service中最好只注入该service对应的dao 若需要用到其他service的dao方法 则注入其他的service 而非直接注入其他service对应的dao
  • 你需要知道的 7 个 Vue3 技巧

    VNode 钩子 在每个组件或html标签上 我们可以使用一些特殊的 文档没写的 钩子作为事件监听器 这些钩子有 onVnodeBeforeMount onVnodeMounted onVnodeBeforeUpdate onVnodeUp
  • 腾讯计费平台部分布式MySQL数据库TDSQL架构分析

    allowtransparency true frameborder 0 scrolling no src http hits sinajs cn A1 weiboshare html url http 3A 2F 2Fwww csdn n
  • STM32 超声波模块测距

    目录 HC SR04模块 HC SR04超声波测距模块工作原理 HC SR04超声波模块的时序触发图 思路步骤 代码 实验结果 总结 HC SR04模块 HC SR04超声波测距模块工作原理 1 采用IO口TRIG触发测距 给至少10us的