找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2790|回复: 4
打印 上一主题 下一主题
收起左侧

超声波测距误差有点大该怎么解决

[复制链接]
跳转到指定楼层
楼主
仿真结果有2cm的误差,到实物就有10cm左右的误差,这该怎么解决,急需,
代码如下:


#include <reg52.h>
#include <intrins.h>
#define uchar unsigned char
#define uint  unsigned int

#include "lcd1602.h"
#include "eepom52.h"
sbit c_send   = P2^7;                //超声波发射
sbit c_recive = P2^6;                //超声波接收
uchar flag_hc_value;        //超声波中间变量


bit flag_300ms ;


bit bSetFlag = 0;     //设置标志位



long distance;                //距离
uint set_d;                    //距离
bit flag_csb_juli;        //超声波超出量程
uint  flag_time0;     //用来保存定时器0的时候的

uchar a_a;


/******************把数据保存到单片机内部eepom中******************/
void write_eepom()
{
        SectorErase(0x2000);
        byte_write(0x2000, set_d % 256);
        byte_write(0x2001, set_d / 256);
        byte_write(0x2058, a_a);       
}

/******************把数据从单片机内部eepom中读出来*****************/
void read_eepom()
{
        set_d  = byte_read(0x2001);
        set_d <<= 8;
        set_d  |= byte_read(0x2000);
        a_a      = byte_read(0x2058);
}

/**************开机自检eepom初始化*****************/
void init_eepom()
{
        read_eepom();                //先读
        if(a_a != 1)                //新的单片机初始单片机内问EEPOM
        {
                set_d = 100;
                a_a = 1;
                write_eepom();
        }       
}

/***********************1ms延时函数*****************************/
void delay_1ms(uint q)
{
        uint i,j;
        for(i=0;i<q;i++)
                for(j=0;j<120;j++);
}

void  delay_us(unsigned int us)
{
        while(us--)
        {
                _nop_();
        }
}


/***********************处理距离函数****************************/
void smg_display()
{
        yujing[0] = distance % 10;       
        yujing[1] = distance / 10 % 10;       
        yujing[2] = distance / 100 % 10;       
}

void delay()
{
        _nop_();                            //执行一条_nop_()指令就是1us
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();
        _nop_();  
        _nop_();
        _nop_();
        _nop_();  
        _nop_();
        _nop_();
        _nop_();  
}


/*********************超声波测距程序*****************************/
void send_wave()
{
        c_send = 1;                           //10us的高电平触发
        delay();
        c_send = 0;         
        TH0 = 0;                          //给定时器0清零
        TL0 = 0;
        TR0 = 0;                                  //关定时器0定时
        flag_hc_value = 0;
        while(!c_recive);                  //当c_recive为零时等待
        TR0=1;
        while(c_recive)                      //当c_recive为1计数并等待
        {
                flag_time0 = TH0 * 256 + TL0;
                if((flag_hc_value > 1) || (flag_time0 > 65000))      //当超声波超过测量范围时,显示3个888
                {
                        TR0 = 0;
                        flag_csb_juli = 2;
                        distance = 888;
                        flag_hc_value = 0;
                        break ;               
                }
                else
                {
                        flag_csb_juli = 1;       
                }
        }
        if(flag_csb_juli == 1)
        {       
                TR0=0;                                                         //关定时器0定时
                distance = TH0;                                      //读出定时器0的时间
                distance = distance * 256 + TL0;
                distance +=( flag_hc_value * 65536);//算出超声波测距的时间         得到单位是ms
                distance *= 0.017;               // 0.017 = 340M / 2 = 170M = 0.017M 算出来是米
                if(distance > 350)                                 //距离 = 速度 * 时间
                {       
                        distance = 888;                                 //如果大于3.8m就超出超声波的量程
                }
        }  
}


/*********************定时器0、定时器1初始化******************/
void time_init()          
{
        EA  = 1;                   //开总中断
        TMOD = 0X11;          //定时器0、定时器1工作方式1
        ET0 = 1;                  //开定时器0中断
        TR0 = 1;                  //允许定时器0定时
        ET1 = 1;                  //开定时器1中断
        TR1 = 1;                  //允许定时器1定时       
}


uchar value = 10;
void main()
{
        send_wave();        //测距离函数
        smg_display();        //处理距离显示函数
        time_init();
        init_1602();
        init_eepom();
       
        while(1)
        {
               
                if(flag_300ms == 1)
                {
                        flag_300ms = 0;
                        if(bSetFlag == 0)
                        {
                                if(++value > 10)                //3秒钟自动测量一次
                                {
                                        value = 0;
                                       
                                        send_wave();        //测距离函数
                                        smg_display();         //处理距离放入数组
                                        write_string(2,1,"Height:");
                                        write_sfm3_csb(2,8,distance);
                                       
                                       
                                }
                                       
                        }
                }
        }
}


/*********************定时器0中断服务程序 用做超声波测距的************************/
void time0_int() interrupt 1  
{                                                  
        flag_hc_value ++;                 //        TH0 TL0 到65536后溢出中断
}                  

/*********************定时器1中断服务程序************************/
void time1_int() interrupt 3
{       
        static uchar value;                         //定时10ms中断一次
        TH1 = 0xf8;
        TL1 = 0x30;     //2ms
        value++;
        if(value >= 150)
        {
                value = 0;
                flag_300ms = 1;
        }

}




6RJYGY5DHRJPAFHF9IY%STR.png (28.21 KB, 下载次数: 41)

6RJYGY5DHRJPAFHF9IY%STR.png
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:97554 发表于 2021-4-26 11:36 | 只看该作者
定时器,频率,距离=声速 X 时间

距离不准就得找单位时间是否准确,这个时间就跟单片机振荡频率相关,因为通过定时器计算时间,那必须考虑晶振频率,12M和20M的晶振用同一个公式那肯定不行的。所以,仔细找一下与计算公式涉及到的数值的获取一定可以找到问题的所在。
                  

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

板凳
ID:332214 发表于 2021-4-26 14:15 | 只看该作者
你这种仿真是无其他干扰的情况下。在实际中,干扰源,有空气湿度 温度等,可以多次采集求平均,在做个校准函数,比如:10cm 采集误差  20cm采集误差  30cm 采集误差 。。。350cm采集误差  求个线性方程(最好是一元二次方程及以上的)。这样求出的结果基本上会很准的了。有条件的,在加个温湿度传感器,依据温湿度的数据,做相应的校准。

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

地板
ID:883698 发表于 2021-4-26 15:47 | 只看该作者
1、程序算法有问题,影响精度
2、定时器误差,延时使用不当,误差变大
3、加入温度补偿,可以提供精度
回复

使用道具 举报

5#
ID:390416 发表于 2021-4-28 00:50 | 只看该作者
直接把定时器的时间 设置为5.78us  表示1mm 的声音距离 直接得出结果,不需要除法了。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表