【STM32F103C8T6】HC-SR04超声波测距模块使用(附有模块文件,适用于江协科技STM32教程)


前言

工程编译环境:Keil5

工程所用硬件:STM32F103C8T6,HC-SR04超声波测距模块,0.96寸OLED,ST_Link,逻辑分析仪(非必需,可方便调试)

文章仅提供代码思路,所使用MCU的外设本文不会有详细介绍。(部分函数来自与江协科技STM32教程)

一、HC-SR04超声波测距模块介绍

1.实物图片:

以下参数来自深圳市艾尔赛科技有限公司的《HC-SR04超声波测距模块说明书》

2.接口定义:

3.模块原理:

4.软件编写思路:

由时序图可知,要想驱动该模块工作,需要通过STM32的GPIO输出一个至少为10us的高电平(在使用时,可以适当延长输出高电平的时间),通过TRGO引脚去告诉HC-SR04该开始工作了。

HC-SR04结束测距后,通过ECHO引脚返回测量结果,由于返回的测量结果形式为高电平持续时间,所以可以利用STM32的外部中断和定时器:通过设置外部中断触发模式为高低电平均可触发,并通过设置一个标志位去记录是否在接收高电平信号,当没有在接收高电平信号时,外部中断触发,打开定时器开始计时,当在接收高电平信号时,外部中断触发,关闭定时器,结束计时。

然后通过图中给的公式:距离(cm) = 所测高电平时间(us) / 58。

5.接线

 VCC5v
TRGOPA2
ECHOPA3
GNDGND

 由于HCSR-04是5v供电,而STM32是3.3v供电,因此需要将VCC引脚接到st-link的5v引脚上进行供电。

6.注意事项:

同时测量周期也至少为60ms(实际使用时,可以适当延长等待时间)

二、程序设计

1.使用STM32F103C8T6读取HC-SR04超声波传感器测量的距离

2.使用0.96寸OLED显示读取的数据(程序设计的OLED驱动和Delay函数均来江协科技

HCSR04Timer.c

使用TIM4作为计时器,预分频系数为72-1,自动重装值为1000-1,可以计算,定时器溢出频率为72MHZ / 72 / 1000 = 1000HZ 也就是一次计数为1us,定时器溢出一次就是1ms。

#include "stm32f10x.h"                  // Device header
void HCSR04_Timer_Init(void)
{
  //打开定时器的时钟总线    
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4,ENABLE);
  //选择内部时钟作为TIM4的时钟    
  TIM_InternalClockConfig(TIM4);
  
  //初始化时基单元    
  TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    
  TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;//时钟不分频
  TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;//时钟向上计数
  TIM_TimeBaseInitStructure.TIM_Period = 1000-1;//计数到1000产生更新事件
  TIM_TimeBaseInitStructure.TIM_Prescaler = 72-1;//72分频
  TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;//自动重装值为0
    
  TIM_TimeBaseInit(TIM4,&TIM_TimeBaseInitStructure); 
  
  //TIM中断配置    
  TIM_ITConfig(TIM4,TIM_IT_Update,ENABLE);    

  //NVIC配置
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//中断优先组选择    
  NVIC_InitTypeDef NVIC_InitStructure;
  
  NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;//选择TIM4中断通道
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;//抢占优先级为2
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;//响应优先级为2
  
  NVIC_Init(&NVIC_InitStructure);
  
  //配置完定时器,先禁止
  TIM_Cmd(TIM4,DISABLE);
}

void HCSR04_Timer_Start(void)
{
  TIM_SetCounter(TIM4,0);//将CNT清零    
  TIM_Cmd(TIM4,ENABLE);//使能TIM4
}

void HCSR04_Timer_Close(void)
{
  TIM_Cmd(TIM4,DISABLE);//失能TIM4
}

HCSR04Timer.h

#ifndef __HCSR04TIMER_H__
#define __HCSR04TIMER_H__

void HCSR04_Timer_Init(void);
void HCSR04_Timer_Start(void);
void HCSR04_Timer_Close(void);
#endif

HCSR04.c

#include "stm32f10x.h"// Device header
#include "Delay.h"//延时函数
#include "HCSR04Timer.h"//HCSR04定时器

#define HCSR04_ECHO_Port GPIOA
#define HCSR04_TRGO_Port GPIOA
#define HCSR04_ECHO_Pin  GPIO_Pin_3
#define HCSR04_TRGO_Pin  GPIO_Pin_2

#define HCSR04_SendTRGO() GPIO_SetBits(HCSR04_TRGO_Port,HCSR04_TRGO_Pin)//向TRGO引脚发送开始信号  
#define HCSR04_OverSend() GPIO_ResetBits(HCSR04_TRGO_Port,HCSR04_TRGO_Pin)//结束向TRGO引脚发送开始信号

uint8_t HCSR04_Flag = 0;//0表示没有在接收回响信号,1表示在接收回响信号中
uint16_t UE_Timers = 0;//TIM4更新时间次数,更新一次为100us

 /**
   * @brief  HCSR04超声波模块初始化,TRGO引脚对应PA4,初始化为推挽输出
   * @brief  ECMO引脚对应PA3,初始化为浮空输入
   * @param  
   * @retval 
   */
void HCSR04_Init(void)
{
  HCSR04_Timer_Init();//HCSR04定时器初始化

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);  //打开GPIOA的时钟总线		
  
  GPIO_InitTypeDef GPIO_InitStructure;//初始化ECHO和TRIG
  //HCSR04_ECHO初始化	
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//引脚设置为浮空输入                                                                                               
  GPIO_InitStructure.GPIO_Pin = HCSR04_ECHO_Pin;//ECHO引脚设置为Pin3
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//引脚速度设置为50MHZ
  GPIO_Init(HCSR04_ECHO_Port,&GPIO_InitStructure);	
  //HCSR04_TRIG初始化
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//推挽输出,强驱动
  GPIO_InitStructure.GPIO_Pin = HCSR04_TRGO_Pin;//TRGO设置为Pin2
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//引脚速度设置为50MHZ
  GPIO_Init(HCSR04_TRGO_Port,&GPIO_InitStructure);
   
  //AFIO映射外部中断引脚	
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);//打开AFIO时钟 
  GPIO_EXTILineConfig(GPIO_PortSourceGPIOA,GPIO_PinSource3);//选择外部中断源和中断通道	

  //外部中断初始化
  EXTI_InitTypeDef EXTI_InitStructure;	
  EXTI_InitStructure.EXTI_Line = EXTI_Line3;//ECHO使用端口PA3
  EXTI_InitStructure.EXTI_LineCmd = ENABLE;//使能
  EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;//选择为中断模式 
  EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising_Falling;//上升沿和下降沿触发	
  
  EXTI_Init(&EXTI_InitStructure);
  
  //NVIC分配外部中断的优先级
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//中断优先组选择
  
  NVIC_InitTypeDef NVIC_InitStructure;
  
  NVIC_InitStructure.NVIC_IRQChannel = EXTI3_IRQn;//选择外部中断3频道
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;//抢占优先级为2
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;//响应优先级为2
  NVIC_Init(&NVIC_InitStructure);
}
/**
   * @brief 放在主函数运行的函数
   * @param
   * @retval HCSR04测得的距离单位为mm
   */
uint32_t HCSR04_Run(void)
{
  	
  UE_Timers = 0;//将计数器溢出次数清零	
  HCSR04_SendTRGO();
  Delay_us(15);//给Trgo引脚15us的高电平信号,开始超声波检测
  HCSR04_OverSend();
  //等待65ms之后读取计数器计时(时间=计数器溢出次数 * 1000 + 读取计数器CNT的值 * 1(单位为us))	
  Delay_ms(65); 
  //根据回响信号返回的高电平时间计算距离		
  uint32_t Distance = (UE_Timers*1000 + TIM_GetCounter(TIM4))/5.8;//结果单位为mm
  return Distance;
}

/**
   * @brief  外部中断函数,用于接收HCSR04的回响信号
   * @param  
   * @retval
   **/
void EXTI3_IRQHandler(void)
{		
  if(EXTI_GetITStatus(EXTI_Line3) == SET)
  {
   //接收来自Echo的上升沿信号,将TIM4打开进行计数	  
   if(HCSR04_Flag == 0)//当Flag=0时,表示没有在接收回响信号
   {
	 HCSR04_Timer_Start();//打开TIM4定时器,开启计时	 
     HCSR04_Flag = 1;//进入接收回响信号状态   
   }
   //接收到来自Echo的下降沿信号,将TIM4关闭
   else if(HCSR04_Flag == 1)//当Flag=1时,表示正在接收回响信号
   {
	 HCSR04_Timer_Close();//关闭TIM4定时器,结束计时
     HCSR04_Flag = 0;//退出接收回响信号状态
   
   }	   
  }
  EXTI_ClearITPendingBit(EXTI_Line3);
}
/**
   * @brief   TIM4对回响信号进行计时
   * @param
   * @retval
   **/
void TIM4_IRQHandler(void)
{
  if(TIM_GetITStatus(TIM4,TIM_IT_Update) == SET)
  {
    UE_Timers++;//每1000us溢出一次 	  
  }	  
  TIM_ClearITPendingBit(TIM4,TIM_IT_Update);
}

HCSR04.h

#ifndef __HCSR04_H__
#define __HCSR04_H__

void  HCSR04_Init(void);
uint32_t HCSR04_Run(void);
#endif

main.c

#include "stm32f10x.h"                  // Device header
#include "OLED.h"
#include "Delay.h"
#include "HCSR04.h"


uint16_t Speed2;
uint16_t i;
uint32_t Distance;
int main(void)
{	

  OLED_Init();//OLED初始化
  HCSR04_Init();//HCSR04初始化
  while(1)
  {
	Distance = 0;
    //多次测量,取平均值,减小误差	  
	for(i=0;i<5;i++)
    {
	  Distance += HCSR04_Run();
	}
    Distance /= 5; 	
    OLED_ShowNum(2,1,Distance,4);

  }

}	  

三、实验效果

虽然测试条件不是很严谨,但是基本的测距功能已实现。

以下是逻辑分析仪抓到的电平信号

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值