|
仿真结果有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;
}
}
|
|