欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136
標題:
單片機+超聲波測距模塊數碼管顯示問題
[打印本頁]
作者:
world365
時間:
2018-8-3 21:56
標題:
單片機+超聲波測距模塊數碼管顯示問題
請問各位大佬,我想讓超聲波測出來的距離顯示在數碼管上,但是我這個程序只在開電的瞬間測出距離并顯示在數碼管上,然后數碼管就一直顯示那個數字,超聲波也不工作了,請問怎么回事?代碼如下。謝謝。。。
#include <reg52.h>
#include <intrins.h>
#include <STDIO.H>
/* 1m所需周期:1000mm/(1微秒的聲波距離mm * 1周期的us時間) => 1000/(0.17*1.0851);
4m = 1000/(0.17*1.0851) * 4 = 21684 */
#define uchar unsigned char
#define iMAX_LEN 21684
#define iMIN_LEN 109
sbit trig = P3^0;
sbit echo = P3^1;
sbit ADDR0 = P1^0;
sbit ADDR1 = P1^1;
sbit ADDR2 = P1^2;
sbit ADDR3 = P1^3;
sbit ENLED = P1^4;
/****************V數碼管模塊*****************************************************************************/
unsigned char code LedChar[] = { //數碼管顯示字符轉換表
0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
unsigned char LedBuff[6] = { //數碼管顯示緩沖區,初值0xFF確保啟動時都不亮
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
},
buf[6],
i = 0,
j;
/*************** ^超聲波模塊*****************************************************************************/
/****************V超聲波模塊*****************************************************************************/
unsigned long miDistance=0, //測距的距離值
distance = 0;
uchar mcDistanceErr=0; //測距錯誤標記(0:正常 / 1:距離太近 /2:超量程)
bit mbDelayOverFlg = 1; //延遲程序的控制標記(默認為延遲結束否則無法進入refreshDistance()函數)
uchar mbDelay10H, mbDelay10L; //測距的小單位時間延遲
/* 定時器函數,T1計數器使用3號中斷
*作用:用于延遲出發測距,每次測距完成后,需要延遲>60ms
*因此當計數器溢出時,time=65.535ms,mbDelayCtlFlg=1 */
void tim1_()interrupt 3
{
TR1 = 0; //T1關閉計數器
mbDelayOverFlg = 1; //設定延遲結束
}
/*啟動超聲波測距模塊,TX保持22us的高電平*/
void StartModule()
{
trig = 1; //控制端置1
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
trig = 0; //控制端置0,等待接收回波
}
/*獲取最近一次測得的距離
*注意:每次成功測距,需要耗時100ms-150ms左右時間*/
unsigned int getDistance(void)
{
return miDistance;
}
/*獲取最近一次的測距狀態
*@return unsigned int 0:正常 / 1:(err)距離太近 /2:(err)超量程*/
unsigned int getDistanceState(void)
{
return mcDistanceErr;
}
/*檢查距離操作(將測得的距離保存在公共變量中)
*備注:本函數調用完成后,需要通過getDistance()或getDistanceState()獲得結果
*注意:每次成功測距,需要耗時100ms-150ms左右時間
*@return 0:完成測距操作 / 1:正在延遲等待下次測距的開始*/
unsigned char refreshDistance(void)
{
unsigned int i; //超量程檢測變量
unsigned int iCycle; //計算總周期
if (1 == mbDelayOverFlg)//判斷是否在延遲期
{
i = iMAX_LEN; //置入最大量程
StartModule(); //發送測距模塊啟動信號
/*此語句的作用:
*沒有收到回波且在N(iMAX_LEN*N)米障礙物信號返回需要的時間前則等待
*(無信號即時返回,防止死循環,阻礙其它程序的執行)*/
while(!echo && i-->0);
//判斷處理結果
if (i>0) //小于N米
{
TR1=1; //收到回波的上邊沿(RX=1),打開計數器
while(echo);//當回波RX=0時,測距結束
TR1=0; //關閉定時器(需要一個時鐘周期)
iCycle = (TH1 * 256 + TL1) + 1; //計算總消耗的周期
TH1=0;
TL1=0;
if (iCycle <= iMIN_LEN)
{
mcDistanceErr = 1;//距離太近
/*距離超近:重啟延遲時間 >10ms,保證上一個聲波回波已經消失
*T0計數器重裝值mbDelay10H, mbDelay10L,在
*InitUltrasonicDistance()初始化函數中生成*/
miDistance = 2;
mbDelayOverFlg = 0; //復位延遲標志
TH1 = mbDelay10H; //重裝計數器高8位
TL1 = mbDelay10L; //重裝計數器低8位
TR1 = 1; //啟動延遲計數器
}
else
{
//(iCycle * 1.0851 * 0.17 / 10) => iCycle * 0.01844670
miDistance = (unsigned int)(iCycle * 184467 / 10000000);//(單位cm)
mcDistanceErr = 0;//測距正常值
/*一次測距完成需要延遲>60ms的時間,保證上一個聲波回波已經消失
* 這兒使用16位計數器的整個空間,> 65.535ms */
mbDelayOverFlg = 0; //復位延遲標志
TH1 = 0; //T0,高位歸0復位
TL1 = 0; //T0,低位歸0復位
TR1 = 1;//打開T0延遲計數器(此時不測距,所以不影響測距的計算)
}
}
else
{
mcDistanceErr = 2; //超量程
miDistance = 400;
}
return 0; //完成測距操作
}
else //正在延遲等待下次測距的開始
return 1;
}
/*************** ^超聲波模塊*****************************************************************************/
//************數碼管顯示
void display(unsigned long sec)
{
buf[0] = sec%10;
buf[1] = sec/10%10;
buf[2] = sec/100%10;
//從最高為開始,遇到0不顯示,遇到非0退出循環
for (j=2; j>=1; j--)
{
if (buf[j] == 0)
LedBuff[j] = 0xFF;
else
break;
}
//將剩余的有效數字位如實轉換
for ( ; j>=0; j--) //for()起始未對j操作,j即保持上個循環結束時的值
{
LedBuff[j] = LedChar[buf[j]];
}
}
void main(void)
{
unsigned int delay_10ms;
ADDR3 = 1;
ENLED = 0;
TMOD |= 0x01;
TH0 = 0xFC; //為T0賦初值0xFC67,定時1ms
TL0 = 0x67;
ET0 = 1; //使能T0中斷
TR0 = 1; //啟動T0
TMOD |= 0x10;
TH1=0; //高位置0
TL1=0; //低位置0
TR1=0; //開始前先關閉計數器
ET1=1; //允許T0中斷
EA = 1;
delay_10ms = (unsigned int)(65536-110592/12); //10ms的延遲提前計算
mbDelay10H = delay_10ms >> 8; //高8位值
mbDelay10L = delay_10ms &0x0F; //低8位值
while(1)
{
//當調用測距函數后,返回為0,表示測距成功,否則測距函數正在延遲中
if (0 == refreshDistance())
{
//當取值有效時,如果與前次值沒變化,則不作更新
if (0 == getDistanceState() && (distance != getDistance()))
{
distance = getDistance();
display(distance);
}
}
}
}
/* 定時器0中斷服務函數 */
void InterruptTimer0() interrupt 1
{
TH0 = 0xFC; //重新加載初值
TL0 = 0x67;
P0 = 0xFF; //顯示消隱
switch (i)
{
case 0: ADDR2=0; ADDR1=0; ADDR0=0; i++; P0=LedBuff[0]; break;
case 1: ADDR2=0; ADDR1=0; ADDR0=1; i++; P0=LedBuff[1]; break;
case 2: ADDR2=0; ADDR1=1; ADDR0=0; i=0; P0=LedBuff[2]; break;
default: break;
}
}
復制代碼
作者:
yzwzfyz
時間:
2018-8-5 10:19
你把程序整理成邏輯框圖就明白了、
歡迎光臨 (http://www.raoushi.com/bbs/)
Powered by Discuz! X3.1