單片機(jī)開(kāi)發(fā)中的一些實(shí)用技巧(上)
二。用軟件擴(kuò)展外部中斷
大家知道,51單片機(jī)的外部中斷只有2個(gè),書(shū)本上曾介紹了一種擴(kuò)展外部中斷源的方法,但是需增加硬件開(kāi)銷(見(jiàn)圖5)。經(jīng)或非門引入外中斷源輸入端(/INT0或/INT1),同時(shí)又連到某I/0口。這樣,每個(gè)“源”都可能引起中斷,在中斷服務(wù)程序中通過(guò)軟件查詢便可確定哪一個(gè)是正在申請(qǐng)的中斷源,其查詢的次序則由中斷源優(yōu)先級(jí)決定,這就可實(shí)現(xiàn)多個(gè)外部中斷源的擴(kuò)展。
圖5
這種方法盡管擴(kuò)展了外部中斷源,但也有不盡人意之處,如設(shè)計(jì)一個(gè)具有8個(gè)中斷源的電路,則需一個(gè)8輸入端的或非門(或門),顯然,對(duì)體積與成本都不利。這里介紹筆者設(shè)計(jì)的擴(kuò)展外部中斷源的方法,由純軟件實(shí)現(xiàn),不添加一個(gè)元件(見(jiàn)圖6)。
圖6
#include P>
static unsigned char data m;//m為全局變量
/*-------延時(shí)子程序-------*/
void delay(unsigned int k)
{
unsigned int i,j;
for(i=0;i
for(j=0;j121;j++)
{;}}
}
/*---外部中斷INT0子程序---*/
void init0()interrupt 0
{
delay(10);//延時(shí)10mS抗抖動(dòng)干擾
if(P3_2==0)
{
EX0=0;//關(guān)INT0中斷
EA=0;//關(guān)總中斷
P3_2=0;//置P3.2為低電平
P2=0xff;//置P2口為全1
m=P2;//讀取P2口狀態(tài)至m
P2=0x00;//恢復(fù)P2口為全0
P3_2=1; //置P3.2為高電平
IT0=1;//置INT0為邊沿觸發(fā)
EX0=1; //開(kāi)INT0中斷
EA=1;} //開(kāi)總中斷
}
/********主程序*********/
void main(void)
{
P2=0x00;// 置P2口為全0
P3_2=1;// 置P3.2為高電平
IT0=1;// 置INT0為邊沿觸發(fā)
EX0=1;// 開(kāi)INT0中斷
EA=1; //開(kāi)總中斷
while(1)//無(wú)限循環(huán)
{
P0=m;//將全局變量m中的內(nèi)容輸出至P0口
P3_0=!P3_0;//P3.0取反,指示程序狀態(tài)
delay(500);//延時(shí)500mS
}
}
程序解釋:無(wú)按鍵按下時(shí),P3.0的發(fā)光管閃亮,作程序狀態(tài)顯示。主程序初始化時(shí),置P2口為全0,置P3.2為高電平,同時(shí)置INT0為邊沿觸發(fā),并開(kāi)放中斷。8個(gè)按鍵的任一個(gè)按下時(shí)都會(huì)引起INT0中斷,進(jìn)入中斷服務(wù)子程序后,首先關(guān)閉中斷,然后置P3.2為低電平,置P2口為全1,再讀取P2口狀態(tài)至m,通過(guò)查詢m的狀態(tài)字即可知道正在申請(qǐng)的中斷源。這里我們采用的方法是將m輸出至P0口點(diǎn)亮LED作指示。退出中斷時(shí),重新開(kāi)放中斷。
評(píng)論