這段時間要用超聲波做一個演示實驗,就是使用超聲波和舵機結合,做一個自動壁障演示實驗。
就是將超聲波接到舵機上,通過轉動舵機來獲取各個方向到小車的距離,從而控制小車運動,避開障礙物,并尋找最佳路徑。整個小車基于航太電子提供的51智能小車,如下圖:
車前面的超聲波模塊就是固定在下面的舵機上面的,實際實驗時需要將顯示屏取下,否則會檔到舵機。
下面簡單說下超聲波模塊:
HC-SR04超聲波測距模塊可提供2cm-400cm的非接觸式距離感測功能,測距精度可達3mm;模塊包括超聲波發(fā)射器、接收器與控制電路,檢測角度為30°。
另外還有超聲波的控制方式:
(1)采用IO口TRIG觸發(fā)測距,給最少10us的高電平信號。
(2)模塊自動發(fā)送8個40khz的方波,自動檢測是否有信號返回;
(3)有信號返回,通過IO口ECHO輸出一個高電平,高電平持續(xù)的時間就是超聲波從發(fā)射到返回的時間。測試距離=(高電平時間*聲速(340M/S))/2
下面說說這個超聲波的缺陷了,由于該超聲波測距本身的缺陷以及該模塊也是市面上比較便宜的模塊,在實際運行時往往達不到要求,特別是小車在運行時整個車子是在震動的,對距離測試十分不利。
當車子在跑動時測出來的距離變動幅度會較大,如果加上舵機的轉動,想實時測到距離是更不可能的了。目前超聲波測量周期建議是100ms,在100ms的時間里,超聲波在車上的變動還是比較大的,所以在測量距離的時候,盡量讓車子停下來,而且舵機停止轉動。
另外還有一個大問題,就是前面的障礙物與超聲波不是正對著,而是呈一個較大的角度時,測出來的距離也是不準的。這種現(xiàn)象體現(xiàn)在當小車與墻斜著跑過去時會直接撞上去,顯然是完全沒檢測到墻面。根據(jù)示波器查看結果,當相對正對超聲波傾斜角度小于約30°時,還是可以測出來的,當變得更大時,就會出現(xiàn)回響電平突然變得很長的情況,這種時候也會有測量比較接近實際的時候,所以在這里需要做濾波處理?;仨懶盘栆词墙咏鼘嶋H,要么是很長,這樣的情況是很好判斷的。
下面是我基于51單片機平臺做的超聲波讀取的方法,暫沒考慮單片機性能浪費的問題,用到了while等待。觸發(fā)信號輸出以及回響信號計數(shù)采用了定時器T2
如下:
初始化函數(shù),對T2 初始化
void UltraSoundInit()
{
Trig = 0;
TH2 = RCAP2H = 0;
TL2 = RCAP2L = 0;
TR2 = 0;//關閉定時器
ET2=1; //允許T2中斷
}
然后就是T2的中斷函數(shù),當定時器溢出時進入,可以用來判斷輸出的回響信號是否過長,當超聲波模塊異常時也可用來跳出while死循環(huán)
/*******************************************************************************
* 函 數(shù) 名 :Timer0Int
* 函數(shù)功能 :定時器0中斷服務函數(shù)
* 輸 入 :無
* 輸 出 :無
*******************************************************************************/
void Timer2Int() interrupt 5 // 定時器2中斷是5號,當定時器2發(fā)生溢出時說明量測的距離過遠或超聲波本次測量異常
{
TF2 = 0;
overflow_count++;
TH2 = 0;
TL2 = 0;
if(overflow_count == 2)
{
status = 5;//超時
}
}
下面就是測量的方法,比較簡單,當測量失敗就讓那個全局變量為0,注意:超聲波是有最小測量距離的,一般障礙物理超聲波2cm以內時,測得距離就不準了,所以測量距離不可能為0
void GetDistance()
{
//發(fā)送觸發(fā)信號
Trig = 1;
status = 1;
TH2 = 0;
TL2 = 0;
TR2 = 1;//打開定時器
while(TL2 < 42);//延時超過10us
status = 2;
Trig = 0;
TR2 = 0;
TH2 = 0;
TL2 = 0;
overflow_count = 0;
TR2 = 1;
while(Echo == 0)//等待回向信號起始位置
{
if(status == 5)
{
status = 0;
distance_cm = 0;
return;//本次失敗
}
}
TR2 = 0;//清空計數(shù)
TH2 = 0;
TL2 = 0;
overflow_count = 0;
TR2 = 1;
while(Echo == 1)//開始計算長度
{
if(status == 5)
{
status = 0;
distance_cm = 0;
TR2 = 0;
return;//本次失敗
}
}
dis_count =overflow_count*65536 + TH2*256 + TL2;
TR2 = 0;
distance_cm = (unsigned int)(((long)(dis_count) * 34)/8000);//聲速 dis_count*(1/(FOSC/12))*Vsound*100/2
status = 0;//準備下次發(fā)送
}
下面就是要調用上面函數(shù)的方法,注意這個函數(shù)執(zhí)行間隔至少是100ms(其實75ms的時候也可以)
void GetDistanceDelay()
{
distance_cm = 0;
while(distance_cm == 0)
{
GetDistance();
if(distance_cm != 0)
{
return;
}
Delayms(100);
}
}
這樣做雖然浪費了很多處理器性能,但比起獲得錯誤數(shù)據(jù)還是要好些。在這個實驗中,除了車子前行時沒100ms檢測一次距離,在其他時候都是要等車子停止時才測距的,另外使用超聲波做這個事本來就是比較勉強的,如果想要更好的效果,建議搭配其他傳感器如光電對管壁障模塊,用這個模塊來檢測斜邊的障礙物還是很有效的,也就是遠處交給超聲波處理,近處交給壁障模塊處理。