历史上的今天
返回首页

历史上的今天

今天是:2025年01月02日(星期四)

正在发生

2020年01月02日 | stm32与HC-SR04超声波传感器测距

2020-01-02 来源:eefocus

首先,先来看一下这个模块的基本功能和原理。


HC-SR04超声波测距模块可提供2cm-400cm的非接触式距离感测功能,测距精度可达高到3mm;模块包括超声波发射器、接收器与控制电路。像智能小车的测距以及转向,或是一些项目中,常常会用到。智能小车测距可以及时发现前方的障碍物,使智能小车可以及时转向,避开障碍物。


注意是5v输入,但是我用stm32 的3.3v输入也是没有问题的。

二.工作原理

      1.给超声波模块接入电源和地。
      2.给脉冲触发引脚(trig)输入一个长为20us的高电平方波

      3.输入方波后,模块会自动发射8个40KHz的声波,与此同时回波引脚(echo)端的电平会由0变为1;(此时应该启动定时器计时)
      4.当超声波返回被模块接收到时,回波引 脚端的电平会由1变为0;(此时应该停止定时器计数),定时器记下的这个时间即为超声波由发射到返回的总时长。
      5.根据声音在空气中的速度为344米/秒,即可计算出所测的距离。

      要学习和应用传感器,学会看懂传感器的时序图是很关键的,所以我们来看一下HC-SR04的时序触发图。

    

 

我们来分析一下这个时序图,先由触发信号启动HC-RS04测距模块,也就是说,主机要先发送至少10us的高电平,触发HC-RS04,模块内部发出信号是传感器自动回应的,我们不用去管它。输出回响信号是我们需要关注的。信号输出的高电平就是超声波发出到重新返回接收所用的时间。用定时器,可以把这段时间记录下来,算出距离,别忘了结果要除于2,因为总时间是发送和接收的时间总和。


下面是亲测可用的驱动程序。


芯片型号为stm32f103zet6,超声波测距后通过串口打印到电脑上面。

驱动和测距;

//超声波测距


#include "hcsr04.h"

 

#define HCSR04_PORT     GPIOB

#define HCSR04_CLK      RCC_APB2Periph_GPIOB

#define HCSR04_TRIG     GPIO_Pin_5

#define HCSR04_ECHO     GPIO_Pin_6


#define TRIG_Send  PBout(5) 

#define ECHO_Reci  PBin(6)


u16 msHcCount = 0;//ms计数


void Hcsr04Init()

{  

    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;     //生成用于定时器设置的结构体

    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(HCSR04_CLK, ENABLE);

     

        //IO初始化

    GPIO_InitStructure.GPIO_Pin =HCSR04_TRIG;       //发送电平引脚

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;//推挽输出

    GPIO_Init(HCSR04_PORT, &GPIO_InitStructure);

    GPIO_ResetBits(HCSR04_PORT,HCSR04_TRIG);

     

    GPIO_InitStructure.GPIO_Pin =   HCSR04_ECHO;     //返回电平引脚

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入

    GPIO_Init(HCSR04_PORT, &GPIO_InitStructure);  

        GPIO_ResetBits(HCSR04_PORT,HCSR04_ECHO);    

     

            //定时器初始化 使用基本定时器TIM6

        RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6, ENABLE);   //使能对应RCC时钟

        //配置定时器基础结构体

        TIM_DeInit(TIM2);

        TIM_TimeBaseStructure.TIM_Period = (1000-1); //设置在下一个更新事件装入活动的自动重装载寄存器周期的值         计数到1000为1ms

        TIM_TimeBaseStructure.TIM_Prescaler =(72-1); //设置用来作为TIMx时钟频率除数的预分频值  1M的计数频率 1US计数

        TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1;//不分频

        TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式

        TIM_TimeBaseInit(TIM6, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位         

        

        TIM_ClearFlag(TIM6, TIM_FLAG_Update);   //清除更新中断,免得一打开中断立即产生中断

        TIM_ITConfig(TIM6,TIM_IT_Update,ENABLE);    //打开定时器更新中断

        hcsr04_NVIC();

    TIM_Cmd(TIM6,DISABLE);     

}



//tips:static函数的作用域仅限于定义它的源文件内,所以不需要在头文件里声明

static void OpenTimerForHc()        //打开定时器

{

        TIM_SetCounter(TIM6,0);//清除计数

        msHcCount = 0;

        TIM_Cmd(TIM6, ENABLE);  //使能TIMx外设

}

 

static void CloseTimerForHc()        //关闭定时器

{

        TIM_Cmd(TIM6, DISABLE);  //使能TIMx外设

}

 

 

 //NVIC配置

void hcsr04_NVIC()

{

            NVIC_InitTypeDef NVIC_InitStructure;

            NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

    

            NVIC_InitStructure.NVIC_IRQChannel = TIM6_IRQn;             //选择串口1中断

            NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;  //抢占式中断优先级设置为1

            NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;         //响应式中断优先级设置为1

            NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;        //使能中断

            NVIC_Init(&NVIC_InitStructure);

}



//定时器6中断服务程序

void TIM6_IRQHandler(void)   //TIM3中断

{

        if (TIM_GetITStatus(TIM6, TIM_IT_Update) != RESET)  //检查TIM3更新中断发生与否

        {

                TIM_ClearITPendingBit(TIM6, TIM_IT_Update  );  //清除TIMx更新中断标志 

                msHcCount++;

        }

}

 


//获取定时器时间

u32 GetEchoTimer(void)

{

        u32 t = 0;

        t = msHcCount*1000;//得到MS

        t += TIM_GetCounter(TIM6);//得到US

          TIM6->CNT = 0;  //将TIM2计数寄存器的计数值清零

                Delay_Ms(50);

        return t;

}

 


//一次获取超声波测距数据 两次测距之间需要相隔一段时间,隔断回响信号

//为了消除余震的影响,取五次数据的平均值进行加权滤波。

float Hcsr04GetLength(void )

{

        u32 t = 0;

        int i = 0;

        float lengthTemp = 0;

        float sum = 0;

        while(i!=5)

        {

        TRIG_Send = 1;      //发送口高电平输出

        Delay_Us(20);

        TRIG_Send = 0;

        while(ECHO_Reci == 0);      //等待接收口高电平输出

            OpenTimerForHc();        //打开定时器

            i = i + 1;

            while(ECHO_Reci == 1);

            CloseTimerForHc();        //关闭定时器

            t = GetEchoTimer();        //获取时间,分辨率为1US

            lengthTemp = ((float)t/58.0);//cm

            sum = lengthTemp + sum ;

        

    }

        lengthTemp = sum/5.0;

        return lengthTemp;

}



/*:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::

** 函数名称: Delay_Ms_Ms

** 功能描述: 延时1MS (可通过仿真来判断他的准确度)            

** 参数描述:time (ms) 注意time<65535

:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::*/

void Delay_Ms(uint16_t time)  //延时函数

    uint16_t i,j;

    for(i=0;i          for(j=0;j<10260;j++);

}

/*:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::

** 函数名称: Delay_Ms_Us

** 功能描述: 延时1us (可通过仿真来判断他的准确度)

** 参数描述:time (us) 注意time<65535                 

:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::*/

void Delay_Us(uint16_t time)  //延时函数

    uint16_t i,j;

    for(i=0;i          for(j=0;j<9;j++);

}


但是关于USART的函数我就不往上写了,这个简单的串口打印大家应该都会写。下面简单贴一下我的主函数吧。

/*

教训:实验前一定要检查引脚连接是否正确,万不可搞错,不然又要烧坏芯片!!!!


*/


#include "hcsr04.h"

#include "chao_usart.h"


int main()

{

    

        float length;

        

        GPIO_cfg();

      NVIC_cfg();

        USART_cfg();    

        printf("串口初始化成功!n");

    

        Hcsr04Init();    

        printf("超声波初始化成功!n");//测试程序是否卡在下面两句上面


        length = Hcsr04GetLength();

        printf("距离为:%.3fn",length);

    

    

}

实验结果:

 

好了,其实这个模块很简单,但是要是把他用的很好的话还是比较困难的,比如用超声波做一个四轴定高的程序,还是有一定的挑战性的。

写这篇博客的目的不仅仅是介绍这个模块的使用,其实这种使用介绍网上一搜一大把,我只是想纪录一下,我在做这个模块的时候遇到的一些其他的问题。

其中有一个小插曲,就是当吧写好的程序烧进去之后,运行时总是出现每次返回一个同样的比正常值小的多的数据,比如说0.034cm,这明显是一个错误的数据,但是刚开始的时候,不知道为什么

总是这样,多次复位从新上电总是这一个数据。让我很是苦恼。但是幸运的是,在这样的情况中间,他又会有时出现一两个正常的的数据,让你有点摸不着头脑。

上网查了一下才慢慢明白,这种现象叫做“余震”,网上关于余震的解释大致有三种:

  1、探头的余震。即使是分体式的,发射头工作完后还会继续震一会,这是物理效应,也就是余震。这个余震信号也会向外传播。如果你的设计是发射完毕后立刻切换为接收状态(无盲区),那么这个余震波会通过壳体和周围的空气,直接到达接收头、干扰了检测(注:通常的测距设计里,发射头和接收头的距离很近,在这么短的距离里超声波的检测角度是很大的,可达180度)。
  2、壳体的余震。就像敲钟一样,能量仍来自发射头。发射结束后,壳体的余震会直接传导到接收头,当然这个时间很短,但已形成了干扰。另外,在不同的环境温度下,壳体的硬度和外形会有所变化,其余震有时长、有时短、有时干扰大、有时干扰小,这是设计工业级产品时必须要考虑的问题。
  3、电路串扰。超声波发射时的瞬间电流很大,例如某种工业级连续测距产品瞬间电流会有15A,通常的产品也能达到1A,瞬间这么大的电流会对电源有一定影响,并干扰接收电路。通过改善电源设计可以缓解这种情况,但在低成本设计中很难根除。所以每次发射完毕,接收电路还需要一段时间稳定工作状态。在此期间,其输出的信号很难使用。


消除上述现象的方法之一就是在检测的时候多次循环检测,取平均值,也就是加权平均滤波,一个简单的滤波处理。就是下面这一段:

        int i = 0;

        float lengthTemp = 0;

        float sum = 0;

        while(i!=5)

        {

        TRIG_Send = 1;      //发送口高电平输出

        Delay_Us(20);

        TRIG_Send = 0;

        while(ECHO_Reci == 0);      //等待接收口高电平输出

            OpenTimerForHc();        //打开定时器

            i = i + 1;

            while(ECHO_Reci == 1);

            CloseTimerForHc();        //关闭定时器

            t = GetEchoTimer();        //获取时间,分辨率为1US

推荐阅读

史海拾趣

ELMOS公司的发展小趣事

1999年,宝马拥有ELMOS公司股份,为ELMOS参与汽车电子化提供了难得的机遇。自此,ELMOS与宝马建立了紧密的合作关系。通过与宝马的深入合作,ELMOS不仅获得了更多的技术支持和市场资源,还积累了丰富的行业经验。这种紧密的合作关系使得ELMOS在汽车电子领域取得了显著的进展,进一步巩固了其在行业中的领先地位。

Chenmoun Enterprise Ltd公司的发展小趣事

随着公司业务的不断扩张,Chenmoun Enterprise Ltd开始实施全球化战略布局。公司在全球范围内设立了研发中心和生产基地,充分利用各地的资源优势和人才优势,提升产品的竞争力。同时,公司积极开拓国际市场,与多家国际知名企业建立了紧密的合作关系,实现了业务的快速增长。

Andersen Laboratories Inc公司的发展小趣事

随着LED背光液晶电视市场的快速发展,AnalogicTech敏锐地捕捉到了这一机遇。公司推出了直接式和边缘式LED背光驱动器系列,显著增强了LED背光液晶电视的能效和用户体验。这一创新举措不仅为AnalogicTech赢得了市场份额,也进一步巩固了其在电子行业中的领先地位。

Discrete Semiconductor Industries公司的发展小趣事

“创新动力”是Discrete Semiconductor Industries公司中一家以创新驱动发展的典范。该公司注重研发投入,积极引进和培养科技人才,建立了完善的研发体系。通过不断的技术创新和产品升级,创新动力不断推出具有市场竞争力的新产品。同时,公司还积极探索新的应用领域和市场机会,不断拓展业务范围。这种创新驱动的发展战略使得创新动力在激烈的市场竞争中保持了强劲的发展势头。

CINCH公司的发展小趣事

CINCH公司作为一家在电子行业颇具影响力的企业,始终致力于技术研发和产品创新。某年,公司研发团队成功开发出一种新型的连接器技术,该技术不仅提高了数据传输速度,还增强了设备的稳定性和耐用性。这一技术突破迅速引起了市场的关注,CINCH公司趁热打铁,推出了一系列基于新技术的产品,迅速占领了市场份额,实现了业务的快速增长。

BAHCO公司的发展小趣事

在市场竞争日益激烈的背景下,BAHCO选择了与同样拥有悠久历史的美国Snap-on公司进行合作。这次强强联合不仅加强了双方在技术、市场等方面的互补优势,更为BAHCO打开了更广阔的市场空间。通过Snap-on的全球销售网络,BAHCO的产品得以进入更多国家和地区,进一步提升了公司的品牌影响力和市场竞争力。

问答坊 | AI 解惑

组装好仪器后液晶屏花屏

我的仪器组装好后,就是把板子和液晶装到外壳后,液晶在过一段时间后就会花屏,不装到外壳里就没事,弄了好长时间了也不知道怎么回事,请大家帮忙。…

查看全部问答>

WinCE5.0挂接IDE硬盘的疑惑?

WinCE5.0挂接IDE硬盘的疑惑? 有个项目,想使用IDE44pin接口挂接最少64GB的外部存储器,现有如下问题和大家讨论: 1.工作电压: 现在手头上的IDE电路是直接从S3C2440上IO口引出,鉴于3.3V的IO电压,似乎不能正常驱 ...…

查看全部问答>

本人大四学生

已经学习了c51单片机,想问下各位老师,如果以后要从事单片机开发的话,需要再学习哪些大学未传授的知识,谢谢…

查看全部问答>

求binArrayStart和binArrayEnd的定义的线索

问题描述:         大家好。我最近在修改Boot程序。在bootinit.c文件中有如下定义: IMPORT UCHAR        binArrayStart [];                    & ...…

查看全部问答>

问个mfc应用的问题??

在dialog中有个tab control,在每个tab的dialog中有个group box, group box中有个按钮。 现在的现实结果是tab control和group box都显示正常,按钮没有显示,什么原因? app方面菜鸟一个,期待大家指点…

查看全部问答>

单片机的问题

提高单片机 WR,RD 驱动能力一般用什么芯片啊…

查看全部问答>

【芯币兑换】AVR单片机学习板及编程器最详细元器件清单

一、USB型51/AVR单片机编程器元器件清单   二、AVR ATmega16单片机学习板元器件清单           [ 本帖最后由 tiankai001 于 2010-8-8 15:35 编辑 ]…

查看全部问答>

一个2407产生对称PWM的问题````

因为刚学DSP,做的是用2407产生6路3对对称PWM,一开始摸索的时候程序老是编译有问题,今天才把程序搞好``却发现我的板子出了问题,因为编的第一个程序所以没有把挖想放上来大家帮我看看这样写是不是能产生6路3对对称PWM``` 主程序如下 #include \\" ...…

查看全部问答>

求一可用的串口通讯程序

折腾了好几天,实在弄不出来了,哪位有好使的串口程序赏一个,谢谢了。…

查看全部问答>

基于LED点阵显示屏9路抢答器的设计制作程序

基于LED点阵显示屏9路抢答器的设计制作程序…

查看全部问答>