第25節(jié):用LED燈和按鍵來模擬工業(yè)自動化設(shè)備的運動控制
前面三節(jié)講了獨立按鍵控制跑馬燈的各種狀態(tài),這一節(jié)我們要做一個機械手控制程序,這個機械手可以左右移動,最左邊有一個開關(guān)感應(yīng)器,最右邊也有一個開關(guān)感應(yīng)器。它也可以上下移動,最下面有一個開關(guān)感應(yīng)器。左右移動是通過一個氣缸控制,上下移動也是通過一個氣缸控制。而單片機控制氣缸,本質(zhì)上是通過三極管把信號放大,然后控制氣缸上的電磁閥。這個系統(tǒng)機械手驅(qū)動部分的輸出和輸入信號如下:
2個輸出IO口,分別控制2個氣缸。對于左右移動的氣缸,當IO口為0時往左邊跑,當IO口為1時往右邊跑。對于上下移動的氣缸,當IO口為0時往上邊跑,當IO口為1時往下邊跑。
3個輸入IO口,分別檢測3個開關(guān)感應(yīng)器。感應(yīng)器沒有被觸發(fā)時,IO口檢測為高電平1。被觸發(fā)時,IO口檢測為低電平0。
這一節(jié)繼續(xù)要教會大家兩個知識點:
第一點:如何用軟件進行開關(guān)感應(yīng)器的抗干擾處理。
第二點:如何用Switch語句搭建工業(yè)自動控制的程序框架。還是那句話,我們只要以Switch語句為支點,再復雜再繁瑣的程序都可以輕松地編寫出來。
具體內(nèi)容,請看源代碼講解。
(1)硬件平臺:基于朱兆祺51單片機學習板。用矩陣鍵盤中的S1鍵作為啟動獨立按鍵,用S5按鍵模擬左邊的開關(guān)感應(yīng)器,用S9按鍵模擬右邊的開關(guān)感應(yīng)器,用S13按鍵模擬下邊的開關(guān)感應(yīng)器。記得把輸出線P0.4一直輸出低電平,模擬獨立按鍵的觸發(fā)地GND。
(2)實現(xiàn)功能:
開機默認機械手在左上方的原點位置。按下啟動按鍵后,機械手從左邊開始往右邊移動,當機械手移動到最右邊時,機械手馬上開始往下移動,最后機械手移動到最右下角的位置時,延時1秒,然后原路返回,一直返回到左上角的原點位置。注意:啟動按鍵必須等機械手處于左上角原點位置時,啟動按鍵的觸發(fā)才有效。
(3)源代碼講解如下:
#include "REG52.H"
#define const_voice_short40 //蜂鳴器短叫的持續(xù)時間
#define const_key_time120 //按鍵去抖動延時的時間
#define const_sensor20 //開關(guān)感應(yīng)器去抖動延時的時間
#define const_1s500//1秒鐘大概的定時中斷次數(shù)
void initial_myself();
void initial_peripheral();
void delay_short(unsigned int uiDelayShort);
void delay_long(unsigned int uiDelaylong);
void left_to_right();//從左邊移動到右邊
void right_to_left(); //從右邊返回到左邊
void up_to_dowm(); //從上邊移動到下邊
void down_to_up(); //從下邊返回到上邊
void run(); //設(shè)備自動控制程序
void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01);
void led_update();//LED更新函數(shù)
void T0_time();//定時中斷函數(shù)
void key_service(); //按鍵服務(wù)的應(yīng)用程序
void key_scan(); //按鍵掃描函數(shù) 放在定時中斷里
void sensor_scan(); //開關(guān)感應(yīng)器軟件抗干擾處理函數(shù),放在定時中斷里。
sbit hc595_sh_dr=P2^3;
sbit hc595_st_dr=P2^4;
sbit hc595_ds_dr=P2^5;
sbit beep_dr=P2^7; //蜂鳴器的驅(qū)動IO口
sbit key_sr1=P0^0; //對應(yīng)朱兆祺學習板的S1鍵
sbit left_sr=P0^1; //左邊的開關(guān)感應(yīng)器 對應(yīng)朱兆祺學習板的S5鍵
sbit right_sr=P0^2; //右邊的開關(guān)感應(yīng)器 有對應(yīng)朱兆祺學習板的S9鍵
sbit down_sr=P0^3; //下邊的開關(guān)感應(yīng)器 對應(yīng)朱兆祺學習板的S13鍵
sbit key_gnd_dr=P0^4; //模擬獨立按鍵的地GND,因此必須一直輸出低電平
unsigned char ucKeySec=0; //被觸發(fā)的按鍵編號
unsigned intuiKeyTimeCnt1=0; //按鍵去抖動延時計數(shù)器
unsigned char ucKeyLock1=0; //按鍵觸發(fā)后自鎖的變量標志
unsigned char ucLeftSr=0;//左邊感應(yīng)器經(jīng)過軟件抗干擾處理后的狀態(tài)標志
unsigned char ucRightSr=0;//右邊感應(yīng)器經(jīng)過軟件抗干擾處理后的狀態(tài)標志
unsigned char ucDownSr=0;//下邊感應(yīng)器經(jīng)過軟件抗干擾處理后的狀態(tài)標志
unsigned intuiLeftCnt1=0;//左邊感應(yīng)器軟件抗干擾所需的計數(shù)器變量
unsigned intuiLeftCnt2=0;
unsigned intuiRightCnt1=0;//右邊感應(yīng)器軟件抗干擾所需的計數(shù)器變量
unsigned intuiRightCnt2=0;
unsigned intuiDownCnt1=0; //下邊軟件抗干擾所需的計數(shù)器變量
unsigned intuiDownCnt2=0;
unsigned intuiVoiceCnt=0;//蜂鳴器鳴叫的持續(xù)時間計數(shù)器
unsigned char ucLed_dr1=0; //代表16個燈的亮滅狀態(tài),0代表滅,1代表亮
unsigned char ucLed_dr2=0;
unsigned char ucLed_dr3=0;
unsigned char ucLed_dr4=0;
unsigned char ucLed_dr5=0;
unsigned char ucLed_dr6=0;
unsigned char ucLed_dr7=0;
unsigned char ucLed_dr8=0;
unsigned char ucLed_dr9=0;
unsigned char ucLed_dr10=0;
unsigned char ucLed_dr11=0;
unsigned char ucLed_dr12=0;
unsigned char ucLed_dr13=0;
unsigned char ucLed_dr14=0;
unsigned char ucLed_dr15=0;
unsigned char ucLed_dr16=0;
unsigned char ucLed_update=1;//刷新變量。每次更改LED燈的狀態(tài)都要更新一次。
unsigned char ucLedStatus16_09=0; //代表底層74HC595輸出狀態(tài)的中間變量
unsigned char ucLedStatus08_01=0; //代表底層74HC595輸出狀態(tài)的中間變量
unsigned intuiRunTimeCnt=0;//運動中的時間延時計數(shù)器變量
unsigned char ucRunStep=0;//運動控制的步驟變量
void main()
{
initial_myself();
delay_long(100);
initial_peripheral();
while(1)
{
run(); //設(shè)備自動控制程序
led_update();//LED更新函數(shù)
key_service(); //按鍵服務(wù)的應(yīng)用程序
}
}
/* 注釋一:
* 開關(guān)感應(yīng)器的抗干擾處理,本質(zhì)上類似按鍵的去抖動處理。唯一的區(qū)別是:
* 按鍵去抖動關(guān)注的是IO口的一種狀態(tài),而開關(guān)感應(yīng)器關(guān)注的是IO口的兩種狀態(tài)。
* 當開關(guān)感應(yīng)器從原來的1狀態(tài)切換到0狀態(tài)之前,要進行軟件濾波處理過程,一旦成功地
* 切換到0狀態(tài)了,再想從0狀態(tài)切換到1狀態(tài)的時候,又要經(jīng)過軟件濾波處理過程,符合
* 條件后才能切換到1的狀態(tài)。通俗的話來說,按鍵的去抖動從1變成0難,從0變成1容易。
* 開關(guān)感應(yīng)器從1變成0難,從0變成1也難。這里所說的"難"是指要經(jīng)過去抖處理。
*/
void sensor_scan() //開關(guān)感應(yīng)器軟件抗干擾處理函數(shù),放在定時中斷里。
{
if(left_sr==1)//左邊感應(yīng)器是高電平,說明有可能沒有被接觸 對應(yīng)朱兆祺學習板的S5鍵
{
uiLeftCnt1=0; //在軟件濾波中,非常關(guān)鍵的語句?。?!類似按鍵去抖動程序的及時清零
uiLeftCnt2++; //類似獨立按鍵去抖動的軟件抗干擾處理
if(uiLeftCnt2>const_sensor)
{
uiLeftCnt2=0;
ucLeftSr=1; //說明感應(yīng)器確實沒有被接觸
}
}
else //左邊感應(yīng)器是低電平,說明有可能被接觸到了
{
uiLeftCnt2=0; //在軟件濾波中,非常關(guān)鍵的語句?。。☆愃瓢存I去抖動程序的及時清零
uiLeftCnt1++;
if(uiLeftCnt1>const_sensor)
{
uiLeftCnt1=0;
ucLeftSr=0; //說明感應(yīng)器確實被接觸到了
}
}
if(right_sr==1)//右邊感應(yīng)器是高電平,說明有可能沒有被接觸 對應(yīng)朱兆祺學習板的S9鍵
{
uiRightCnt1=0; //在軟件濾波中,非常關(guān)鍵的語句?。?!類似按鍵去抖動程序的及時清零
uiRightCnt2++; //類似獨立按鍵去抖動的軟件抗干擾處理
if(uiRightCnt2>const_sensor)
{
uiRightCnt2=0;
ucRightSr=1; //說明感應(yīng)器確實沒有被接觸
}
}
else //右邊感應(yīng)器是低電平,說明有可能被接觸到了
{
uiRightCnt2=0; //在軟件濾波中,非常關(guān)鍵的語句?。。☆愃瓢存I去抖動程序的及時清零
uiRightCnt1++;
if(uiRightCnt1>const_sensor)
{
uiRightCnt1=0;
ucRightSr=0; //說明感應(yīng)器確實被接觸到了
}
}
if(down_sr==1)//下邊感應(yīng)器是高電平,說明有可能沒有被接觸 對應(yīng)朱兆祺學習板的S13鍵
{
uiDownCnt1=0; //在軟件濾波中,非常關(guān)鍵的語句!??!類似按鍵去抖動程序的及時清零
uiDownCnt2++; //類似獨立按鍵去抖動的軟件抗干擾處理
if(uiDownCnt2>const_sensor)
{
uiDownCnt2=0;
ucDownSr=1; //說明感應(yīng)器確實沒有被接觸
}
}
else //下邊感應(yīng)器是低電平,說明有可能被接觸到了
{
uiDownCnt2=0; //在軟件濾波中,非常關(guān)鍵的語句?。?!類似按鍵去抖動程序的及時清零
uiDownCnt1++;
if(uiDownCnt1>const_sensor)
{
uiDownCnt1=0;
ucDownSr=0; //說明感應(yīng)器確實被接觸到了
}
}
}
void key_scan()//按鍵掃描函數(shù) 放在定時中斷里
{
if(key_sr1==1)//IO是高電平,說明按鍵沒有被按下,這時要及時清零一些標志位
{
ucKeyLock1=0; //按鍵自鎖標志清零
uiKeyTimeCnt1=0;//按鍵去抖動延時計數(shù)器清零,此行非常巧妙,是我實戰(zhàn)中摸索出來的。
}
else if(ucKeyLock1==0)//有按鍵按下,且是第一次被按下
{
uiKeyTimeCnt1++; //累加定時中斷次數(shù)
if(uiKeyTimeCnt1>const_key_time1)
{
uiKeyTimeCnt1=0;
ucKeyLock1=1;//自鎖按鍵置位,避免一直觸發(fā)
ucKeySec=1; //觸發(fā)1號鍵
}
}
}
void key_service() //按鍵服務(wù)的應(yīng)用程序
{
switch(ucKeySec) //按鍵服務(wù)狀態(tài)切換
{
case 1:// 啟動按鍵 對應(yīng)朱兆祺學習板的S1鍵
if(ucLeftSr==0)//處于左上角原點位置
{
ucRunStep=1; //啟動
uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
}
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
}
}
void led_update()//LED更新函數(shù)
{
if(ucLed_update==1)
{
ucLed_update=0; //及時清零,讓它產(chǎn)生只更新一次的效果,避免一直更新。
if(ucLed_dr1==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x01;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0xfe;
}
if(ucLed_dr2==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x02;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0xfd;
}
if(ucLed_dr3==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x04;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0xfb;
}
if(ucLed_dr4==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x08;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0xf7;
}
if(ucLed_dr5==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x10;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0xef;
}
if(ucLed_dr6==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x20;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0xdf;
}
if(ucLed_dr7==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x40;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0xbf;
}
if(ucLed_dr8==1)
{
ucLedStatus08_01=ucLedStatus08_01|0x80;
}
else
{
ucLedStatus08_01=ucLedStatus08_01&0x7f;
}
if(ucLed_dr9==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x01;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0xfe;
}
if(ucLed_dr10==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x02;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0xfd;
}
if(ucLed_dr11==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x04;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0xfb;
}
if(ucLed_dr12==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x08;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0xf7;
}
if(ucLed_dr13==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x10;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0xef;
}
if(ucLed_dr14==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x20;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0xdf;
}
if(ucLed_dr15==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x40;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0xbf;
}
if(ucLed_dr16==1)
{
ucLedStatus16_09=ucLedStatus16_09|0x80;
}
else
{
ucLedStatus16_09=ucLedStatus16_09&0x7f;
}
hc595_drive(ucLedStatus16_09,ucLedStatus08_01);//74HC595底層驅(qū)動函數(shù)
}
}
void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01)
{
unsigned char i;
unsigned char ucTempData;
hc595_sh_dr=0;
hc595_st_dr=0;
ucTempData=ucLedStatusTemp16_09;//先送高8位
for(i=0;i<8;i++)
{
if(ucTempData>=0x80)hc595_ds_dr=1;
else hc595_ds_dr=0;
hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
delay_short(15);
hc595_sh_dr=1;
delay_short(15);
ucTempData=ucTempData<<1;
}
ucTempData=ucLedStatusTemp08_01;//再先送低8位
for(i=0;i<8;i++)
{
if(ucTempData>=0x80)hc595_ds_dr=1;
else hc595_ds_dr=0;
hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
delay_short(15);
hc595_sh_dr=1;
delay_short(15);
ucTempData=ucTempData<<1;
}
hc595_st_dr=0;//ST引腳把兩個寄存器的數(shù)據(jù)更新輸出到74HC595的輸出引腳上并且鎖存起來
delay_short(15);
hc595_st_dr=1;
delay_short(15);
hc595_sh_dr=0; //拉低,抗干擾就增強
hc595_st_dr=0;
hc595_ds_dr=0;
}
void left_to_right()//從左邊移動到右邊
{
ucLed_dr1=1; // 1代表左右氣缸從左邊移動到右邊
ucLed_update=1;//刷新變量。每次更改LED燈的狀態(tài)都要更新一次。
}
void right_to_left() //從右邊返回到左邊
{
ucLed_dr1=0; // 0代表左右氣缸從右邊返回到左邊
ucLed_update=1;//刷新變量。每次更改LED燈的狀態(tài)都要更新一次。
}
void up_to_down() //從上邊移動到下邊
{
ucLed_dr2=1; // 1代表上下氣缸從上邊移動到下邊
ucLed_update=1;//刷新變量。每次更改LED燈的狀態(tài)都要更新一次。
}
void down_to_up() //從下邊返回到上邊
{
ucLed_dr2=0; // 0代表上下氣缸從下邊返回到上邊
ucLed_update=1;//刷新變量。每次更改LED燈的狀態(tài)都要更新一次。
}
void run() //設(shè)備自動控制程序
{
switch(ucRunStep)
{
case 0: //機械手處于左上角原點的位置,待命狀態(tài)。此時觸發(fā)啟動按鍵ucRunStep=1,就觸發(fā)后續(xù)一些列的連續(xù)動作。
break;
case 1: //機械手從左邊往右邊移動
left_to_right();
ucRunStep=2;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
break;
case 2: //等待機械手移動到最右邊,直到觸發(fā)了最右邊的開關(guān)感應(yīng)器。
if(ucRightSr==0)//右邊感應(yīng)器被觸發(fā)
{
ucRunStep=3;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
}
break;
case 3: //機械手從右上邊往右下邊移動,從上往下。
up_to_down();
ucRunStep=4;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
break;
case 4: //等待機械手從右上邊移動到右下邊,直到觸發(fā)了右下邊的開關(guān)感應(yīng)器。
if(ucDownSr==0)//右下邊感應(yīng)器被觸發(fā)
{
uiRunTimeCnt=0;//時間計數(shù)器清零,為接下來延時1秒鐘做準備
ucRunStep=5;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
}
break;
case 5: //機械手在右下邊延時1秒
if(uiRunTimeCnt>const_1s)//延時1秒
{
ucRunStep=6;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
}
break;
case 6: //原路返回,機械手從右下邊往右上邊移動。
down_to_up();
ucRunStep=7;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
break;
case 7: //原路返回,等待機械手移動到最右邊的感應(yīng)開關(guān)
if(ucRightSr==0)
{
ucRunStep=8;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
}
break;
case 8: //原路返回,等待機械手從右邊往左邊移動
right_to_left();
ucRunStep=9;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
break;
case 9: //原路返回,等待機械手移動到最左邊的感應(yīng)開關(guān),表示返回到了原點
if(ucLeftSr==0) //返回到左上角的原點位置
{
ucRunStep=0;//這就是鴻哥傳說中的怎樣靈活控制步驟變量
}
break;
}
}
void T0_time() interrupt 1
{
TF0=0;//清除中斷標志
TR0=0; //關(guān)中斷
sensor_scan(); //開關(guān)感應(yīng)器軟件抗干擾處理函數(shù)
key_scan(); //按鍵掃描函數(shù)
if(uiRunTimeCnt<0xffff) //不要超過最大int類型范圍
{
uiRunTimeCnt++; //延時計數(shù)器
}
if(uiVoiceCnt!=0)
{
uiVoiceCnt--; //每次進入定時中斷都自減1,直到等于零為止。才停止鳴叫
beep_dr=0;//蜂鳴器是PNP三極管控制,低電平就開始鳴叫。
}
else
{
; //此處多加一個空指令,想維持跟if括號語句的數(shù)量對稱,都是兩條指令。不加也可以。
beep_dr=1;//蜂鳴器是PNP三極管控制,高電平就停止鳴叫。
}
TH0=0xf8; //重裝初始值(65535-2000)=63535=0xf82f
TL0=0x2f;
TR0=1;//開中斷
}
void delay_short(unsigned int uiDelayShort)
{
unsigned int i;
for(i=0;i
; //一個分號相當于執(zhí)行一條空語句
}
}
void delay_long(unsigned int uiDelayLong)
{
unsigned int i;
unsigned int j;
for(i=0;i
for(j=0;j<500;j++)//內(nèi)嵌循環(huán)的空指令數(shù)量
{
; //一個分號相當于執(zhí)行一條空語句
}
}
}
void initial_myself()//第一區(qū) 初始化單片機
{
/* 注釋二:
* 矩陣鍵盤也可以做獨立按鍵,前提是把某一根公共輸出線輸出低電平,
* 模擬獨立按鍵的觸發(fā)地,本程序中,把key_gnd_dr輸出低電平。
* 朱兆祺51學習板的S1就是本程序中用到的一個獨立按鍵。
*/
key_gnd_dr=0; //模擬獨立按鍵的地GND,因此必須一直輸出低電平
beep_dr=1; //用PNP三極管控制蜂鳴器,輸出高電平時不叫。
TMOD=0x01;//設(shè)置定時器0為工作方式1
TH0=0xf8; //重裝初始值(65535-2000)=63535=0xf82f
TL0=0x2f;
}
void initial_peripheral() //第二區(qū) 初始化外圍
{
EA=1; //開總中斷
ET0=1; //允許定時中斷
TR0=1; //啟動定時中斷
}
總結(jié)陳詞:
前面花了很多節(jié)內(nèi)容在講按鍵和跑馬燈的關(guān)系,但是一直沒涉及到人機界面,在大多數(shù)的實際項目中,人機界面是必不可少的。人機界面的程序框架該怎么樣寫?欲知詳情,請聽下回分解-----在主函數(shù)while循環(huán)中驅(qū)動數(shù)碼管的動態(tài)掃描程序。
評論