用以下程序做的超聲波測距1602不能實時顯示距離,顯示的距離永遠是第一次次測量的距離,求解
#include <AT89x51.H>
#include <intrins.h>
#define uint unsigned int
#define uchar unsigned char
#define RX P2_6
#define TX P2_7
#define LCM_RW P0_2
#define LCM_RS P0_1
#define LCM_E P0_3
#define LCM_Data P1
#define Busy 0x80
unsigned char code mcustudio[] ={" DONGMUFORYOU "};
unsigned char code Cls[] = {" "};
unsigned char code ASCII[15] = {'0','1','2','3','4','5','6','7','8','9','.','-','M'};
static unsigned char DisNum = 0;
uint time=0;
unsigned long S=0;
bit flag =0;
uchar disbuff[4]={ 0,0,0,0,};
/*********************5ms延時************/
void delay5Ms(void)
{
uint TempCyc = 5552;
while(TempCyc--);
}
/***********400ms延時******************/
void delay400Ms(void)
{
uchar TempCycA = 5;
uint TempCycB;
while(TempCycA--)
{
TempCycB=7269;
while(TempCycB--);
};
}
/*********************延遲****************/
void delayms(uint ms)
{
unsigned char i=100,j;
for(;ms;ms--)
{
while(--i)
{
j=10;
while(--j);
}
}
}
/**************讀狀態**********************************/
uchar readStatusLCM(void)
{
LCM_Data = 0xFF;
LCM_RS = 0;
LCM_RW = 1;
LCM_E = 0;
LCM_E = 0;
LCM_E = 1;
while(LCM_Data & Busy); //檢測忙信號
return(LCM_Data);
}
/*****************讀數據***************************************/
/*uchar readDataLCM(void)
{
LCM_RS = 1;
LCM_RW = 1;
LCM_E = 0;
LCM_E = 0;
LCM_E = 1;
return(LCM_Data);
} */
/******************寫數據*********************************/
void writeDataLCM(uchar WDLCM)
{
readStatusLCM(); //檢測忙
LCM_Data = WDLCM;
LCM_RS = 1;
LCM_RW = 0;
LCM_E = 0; //若晶振速度太高可以在這后加小的延時
LCM_E = 0; //延時
LCM_E = 1;
}
/*****************寫指令*********************************/
void writeCommandLCM(unsigned char WCLCM,BuysC) //BuysC為0時忽略忙檢測
{
if (BuysC)
readStatusLCM(); //根據需要檢測忙
LCM_Data = WCLCM;
LCM_RS = 0;
LCM_RW = 0;
LCM_E = 0;
LCM_E = 0;
LCM_E = 1;
}
/*************初始化*************************************/
void LCMInit(void) //LCM初始化
{
writeCommandLCM(0x38,0); //顯示模式設置,不檢測忙信號
delay5Ms();
writeCommandLCM(0x38,1); //顯示模式設置,開始要求每次檢測忙信號
writeCommandLCM(0x08,1); //關閉顯示
writeCommandLCM(0x01,1); //顯示清屏
writeCommandLCM(0x06,1); // 顯示光標移動設置
writeCommandLCM(0x0c,1); // 顯示關及光標設置
}
//按指定位置顯示一個字符
void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)
{
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
if (Y)
{
X |= 0x40; //當要顯示第二行時地址碼+0x40;
}
X |= 0x80; //算出指令碼
writeCommandLCM(X, 1); //發命令字
writeDataLCM(DData); //發數據
}
/**************顯示字符*************************************/
//按指定位置顯示一串字符
void displayListChar(uchar X, uchar Y, uchar code *DData)
{
uchar ListLength;
ListLength = 0;
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
while (DData[ListLength]>0x19) //若到達字串尾則退出
{
if (X <= 0xF) //X坐標應小于0xF
{
DisplayOneChar(X, Y, DData[ListLength]); //顯示單個字符
ListLength++;
X++;
}
}
}
/************計數*****************************/
void count(void)
{
time=TH0*256+TL0;
TH0=0;
TL0=0;
S=(time*1.7)/100; //算出來是CM
if((S>=700)||flag==1) //當距離超出測量時,范圍顯示“-”
{
flag=0;
DisplayOneChar(0, 1, ASCII[11]);
DisplayOneChar(1, 1, ASCII[10]); //顯示點
DisplayOneChar(2, 1, ASCII[11]);
DisplayOneChar(3, 1, ASCII[11]);
DisplayOneChar(4, 1, ASCII[12]); //顯示M
}
else //當距離未超出距離限制時,正常顯示
{
disbuff[0]=S%1000/100;
disbuff[1]=S%1000%100/10;
disbuff[2]=S%1000%10 %10;
DisplayOneChar(4, 1, ASCII[disbuff[0]]);
DisplayOneChar(5, 1, ASCII[10]); //顯示點
DisplayOneChar(6, 1, ASCII[disbuff[1]]);
DisplayOneChar(7, 1, ASCII[disbuff[2]]);
DisplayOneChar(8, 1, ASCII[12]); //顯示M
}
}
/*************************中斷**************************/
void timer0() interrupt 1 //T0中斷用來計數器溢出,超過測距范圍
{
flag=1; //中斷溢出標志
}
/******************啟動模塊*********************************/
void startModule() //啟動模塊
{
uchar i;
TX=1; //啟動一次模塊
for(i=0;i<20;i++) //延遲 以便于進行傳送數據
{
_nop_();
}
TX=0;
}
/*****************主函數***********************************/
void main()
{
LCMInit(); //LCM初始化
delay5Ms(); //延時片刻(可不要)
displayListChar(0, 0, mcustudio);
displayListChar(0, 1, Cls);
TMOD=0x01; //設T0為方式1,GATE=1;
TH0=0;
TL0=0;
ET0=1; //允許T0中斷
EA=1; //開啟總中斷
while(1)
{
startModule();
while(!RX); //當RX為零時等待
TR0=1; //開啟計數
while(RX); //當RX為1計數并等待
TR0=0; //關閉計數
count(); //計算
delayms(80); //80MS
}
大神們求解!急急急 |