基于K60/K66/LPC的HC-SR04超声波测距(附带代码)

最近在搞超声波,把自己走过的一些坑,和经验分享一下,互相学习,让初学者少走一些弯路,K60代码网上找了一些,结果没有能用的,没办法,自己看讲解,用PIT计时测试成功,在K60和K66上测试可以使用,测距也比较准确,希望能有些帮助

1HC-SR04超声波测距原理

准备 :引出4个排针,连接到单片机的vcc(5V),io口,io口,gnd,用到2个io口,

1.给脉冲触发引脚(Trig)输入一个持续时间>10us的高电平

2.输入方波后,模块会自动发射8个40KHz的声波,与此同时回波引脚(ECOH)端的电平会由0变为1; (输出一个高电平)(此时应该启动定时器计时)
.3.当超声波返回被模块接收到时,接收端的电平会由1变为0(变为低电平);该高电平的持续时间即为超声波往返的时间,(这时停止定时器计数),

4.在单片机里面通过定时器计算Echo高电平的持续时间即可算出超声波往返的距离

5.测距公式:距离=(高电平时间*声速)/2;
在这里插入图片描述

2.调试经验

1注意超声波要接的是5V,不是3.3V,使用下载器时一般都是3.3V供电,这就导致在线调试没有数据,应该给板子供上电,下载器不供电,超声波接到5V电源口,并且检测下是不是5V.

2.测距时不要距离太近,会导致数据不准确

3.超声波代码要尽量精简,不要繁琐,假设你代码过于繁琐,因为超声波测距时间非常短(us单位)可能这次电平已经发生变化,但是你的MCU在经行其他代码运行,可能就检测不到这次信号。

4.要考虑各种因素影响测距稳定,比如高电平检测是否准确,温度原因等,超声波测距的性能与被测物表面材料有很大关系,如毛料、布料对超声波 的反射率很小,会严重影响测量结果。

5.io口任意两个即可,注意不要占用到之前本来使用的,否则会出问题,还有PIT1如果使用,就初始化PIT2进行计时,一个PIT做一种工作

6 测量时间间隔在60MS左右最好

7 测距等待一定的时间,如果超时后依然没有数据返回,就放弃,而进行下一次测量。

超声波最大测距4米,反射路径8米,声速度331米/秒) 8/331=0.02416 //24MS



话不多说直接上代码⬇

#include  "MK60_PIT.h"   
#define TRIG    D9    //超声波发送端
#define ECHG    E12   //超声波接收端
int Distance = 0;   //测量距离
gpio_init(TRIG,GPO,0);    //初始化发送端
gpio_init(ECHG,GPI,0);    //初始化接收端
 /*
 * @file       distance
 * @brief      超声波测距
 * @author      Z小旋
 * @version    v1.0
 * @date       2019-1-17
*/
void distance(void)
{
     uint32 Time = 0;
     gpio_set(TRIG,1);               //产生触发脉冲
     pit_delay_us(pit1,20);       
     gpio_set(TRIG,0);           //产生一个20us的高电平脉冲
        
        
     while(gpio_get (ECHG) == 0);             //等待电平变高,低电平一直等待
     pit_time_start  (pit1); //开始计时
     while(gpio_get(ECHG) == 1)              //等待电平变低,高电平一直等待
     {
       Time++;
       systick_delay(1);  
       if(Time>240000) break; //24MS还没有检测到     
     };               
     timevar = pit_time_get(pit1);    //停止计时,获取计时时间
         //timevar = timevar * 340 /2/10;
     Distance = timevar*(331.4+0.607*10)/2000;  //加上温度补偿
   //  DELAY_MS(60); //延时60MS,测距更精确,可以不用,2.27号:有反映说加上之后测距不精确,注释之后就好了,自行测试
} 

上面代码测距误差在10mm以内,满足正常使用,直接调用函数即可
在这里插入图片描述
下面这个是使用外部触发中断进行的,附上完整代码,直接复制使用即可,如果没有对应的库,完整例程在下方下载,弄到百度云了,CSDN下载还要钱。。。

#include "headfile.h"

#define TRIG A14                //定义超声波触发引脚端口
#define ECHO A13                //定义超声波回响引脚端口

 /*
 * @brief      超声波测距
 * @author      Z小旋
 * @version    v1.0
*/

uint8  flag_mode = 0;//当前运行状态 0:采集结束  1:正在采集  
uint32 dis_time;//定义时间变量 单位微秒
uint32 distance;//定义距离变量 单位毫米

int main(void)
{   
  		gpio_init (TRIG,GPO,0);                                                   //触发引脚初始化 新生板A28
		port_init (ECHO, IRQ_FALLING | PF | ALT1 | PULLUP );                      //接收引脚初始化 新生板E9
		//port_init (ECHO_PIN, IRQ_RISING | PF | ALT1 | PULLUP );
		set_irq_priority(PORTA_IRQn,3);//设置优先级 越低优先级越高
		enable_irq(PORTA_IRQn);//开中断 PIN_INT0_IRQn - PIN_INT7_IRQn
		EnableInterrupts;
   

    //这里是while循环,使用时请封装成函数并在中断中调用,不宜调用过快,否则会很占系统时序
    while(1)
    {
      if(!flag_mode)                   
      {
      
        flag_mode = 1;
        gpio_set(TRIG, 1);
        pit_delay_us(pit1,15);       
        gpio_set(TRIG, 0);
        
        while(!gpio_get(ECHO)); //检测到接收引脚为高电平则开始计时
        pit_time_start  (pit1); //开始计时
        dis_time = 0;               //时间清零
      }
      
      
      //超时检测
      if(20000 <= pit_time_get(pit1))   //如果等待20ms之后还未接收到回响信号则 认为前方无障碍物  
      {
        dis_time = 0;           //时间清零
        distance = 1000;       
        flag_mode = 0;          //采集结束
      }   
      
    }
}




//-------------------------------------------------------------------------------------------------------------------
//  @brief      PROTA中断执行函数
//  @return     void   
//  @since      v1.0
//  Sample usage:               当A口启用中断功能且发生中断的时候会自动执行该函数
//-------------------------------------------------------------------------------------------------------------------
void PORTA_IRQHandler(void)
{
    //清除中断标志第一种方法直接操作寄存器,每一位对应一个引脚
  	PORTA_FLAG_CLR(A13);
      if(flag_mode)
    {
        dis_time = pit_time_get(pit1);        //获取时间
       distance = (int16)(dis_time*(331.4+0.607*10)/20000);  //加上温度补偿
        flag_mode = 0;
    }


}





链接: https://pan.baidu.com/s/1YP8SuujxxRzOxO2cm_KWIQ 提取码: btyv

整理不易,都看到这儿了,点个赞再走呗

STC15F2K60S2是一款基于8位单片机的微控制器,而LED超声波测距模块通常用于测量物体的距离。在使用这款单片机配合超声波传感器进行测距时,你需要编写一段程序来控制脉冲发送、接收回音并计算距离。 以下是一个基本的STC15F2K60S2配合HC-SR04超声波模块的测距代码示例(假设你已经在单片机中配置了相应的IO口): ```c #include "reg52.h" // STC15系列的头文件 #define TRIG_PIN P2_0 // 发射信号的引脚 #define ECHO_PIN P2_1 // 接收回音的引脚 #define MAX_RANGE 400 // 测量范围设定,单位毫米 void delay_us(unsigned int us); int get_distance(); int main(void) { TRISD = 0; // 设置IO口为输入输出模式 while (1) { unsigned int distance = get_distance(); if (distance < MAX_RANGE) { // 显示或处理距离数据 printf("Distance: %d mm\n", distance); } else { printf("Object out of range!\n"); } delay_ms(500); // 每次测距后等待一段时间 } } void delay_us(unsigned int us) { unsigned int i; for (i = us; i > 0; i--) { __nop(); // 单周期延时 } } int get_distance() { unsigned char trigState = 0, echoState = 0; unsigned int pulseStart, pulseEnd, pulseDuration; // 发送触发脉冲 TRIG_PIN = !trigState; delay_us(10); // 高电平持续10us trigState = 1; delay_us(10); // 低电平持续10us // 等待回声 while (TRIG_PIN == 1); // 确保触发结束 pulseStart = GET_US(); while (ECHO_PIN == 0); // 找到第一个下降沿 pulseStart = GET_US(); while (ECHO_PIN == 1); // 找到第二个下降沿,即回声到达 pulseEnd = GET_US(); pulseDuration = pulseEnd - pulseStart; // 计算脉冲时间 // 根据公式计算距离 unsigned int distance = pulseDuration * 0.034 / 2; // HC-SR04默认58mm/cycle return distance; } ```
评论 19
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Z小旋

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值