新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > 第十四節(jié):矩陣鍵盤的單個觸發(fā)

第十四節(jié):矩陣鍵盤的單個觸發(fā)

作者: 時間:2016-11-22 來源:網(wǎng)絡(luò) 收藏
開場白:
上一節(jié)講了按鍵的加速勻速觸發(fā)。這節(jié)開始講矩陣鍵盤單個觸發(fā)。

具體內(nèi)容,請看源代碼講解。

(1)硬件平臺:基于朱兆祺51單片機(jī)學(xué)習(xí)板。。

(2)實(shí)現(xiàn)功能:16個按鍵中,每按一個按鍵都能觸發(fā)一次蜂鳴器發(fā)出“滴”的一聲。

(3)源代碼講解如下:
#include "REG52.H"

#define const_voice_short40 //蜂鳴器短叫的持續(xù)時間

#define const_key_time20 //按鍵去抖動延時的時間

void initial_myself();
void initial_peripheral();
void delay_long(unsigned int uiDelaylong);
void T0_time();//定時中斷函數(shù)
void key_service(); //按鍵服務(wù)的應(yīng)用程序
void key_scan(); //按鍵掃描函數(shù) 放在定時中斷里

sbit key_sr1=P0^0; //第一行輸入
sbit key_sr2=P0^1; //第二行輸入
sbit key_sr3=P0^2; //第三行輸入
sbit key_sr4=P0^3; //第四行輸入

sbit key_dr1=P0^4; //第一列輸出
sbit key_dr2=P0^5; //第二列輸出
sbit key_dr3=P0^6; //第三列輸出
sbit key_dr4=P0^7; //第四列輸出

sbit beep_dr=P2^7; //蜂鳴器的驅(qū)動IO口

unsigned char ucKeyStep=1;//按鍵掃描步驟變量

unsigned char ucKeySec=0; //被觸發(fā)的按鍵編號
unsigned intuiKeyTimeCnt=0; //按鍵去抖動延時計數(shù)器
unsigned char ucKeyLock=0; //按鍵觸發(fā)后自鎖的變量標(biāo)志


unsigned intuiVoiceCnt=0;//蜂鳴器鳴叫的持續(xù)時間計數(shù)器

void main()
{
initial_myself();
delay_long(100);
initial_peripheral();
while(1)
{
key_service(); //按鍵服務(wù)的應(yīng)用程序
}

}

void key_scan()//按鍵掃描函數(shù) 放在定時中斷里
{
/* 注釋一:
*矩陣按鍵掃描的詳細(xì)過程:
*先輸出某一列低電平,其它三列輸出高電平,這個時候再分別判斷輸入的四行,
*如果發(fā)現(xiàn)哪一行是低電平,就說明對應(yīng)的某個按鍵被觸發(fā)。依次分別輸出另外三列
*中的某一列為低電平,再分別判斷輸入的四行,就可以檢測完16個按鍵。內(nèi)部詳細(xì)的
*去抖動處理方法跟我前面講的獨(dú)立按鍵去抖動方法是一樣的。
*/

switch(ucKeyStep)
{
case 1: //按鍵掃描輸出第一列低電平
key_dr1=0;
key_dr2=1;
key_dr3=1;
key_dr4=1;

uiKeyTimeCnt=0;//延時計數(shù)器清零
ucKeyStep++; //切換到下一個運(yùn)行步驟
break;

case 2: //此處的小延時用來等待剛才列輸出信號穩(wěn)定,再判斷輸入信號。不是去抖動延時。
uiKeyTimeCnt++;
if(uiKeyTimeCnt>1)
{
uiKeyTimeCnt=0;
ucKeyStep++; //切換到下一個運(yùn)行步驟
}
break;

case 3:
if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
ucKeyStep++;//如果沒有按鍵按下,切換到下一個運(yùn)行步驟
ucKeyLock=0;//按鍵自鎖標(biāo)志清零
uiKeyTimeCnt=0; //按鍵去抖動延時計數(shù)器清零,此行非常巧妙

}
else if(ucKeyLock==0)//有按鍵按下,且是第一次觸發(fā)
{
if(key_sr1==0&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=1;//觸發(fā)1號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S1鍵
}

}
else if(key_sr1==1&&key_sr2==0&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=5;//觸發(fā)5號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S5鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==0&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=9;//觸發(fā)9號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S9鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==0)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=13;//觸發(fā)13號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S13鍵
}

}

}
break;

case 4: //按鍵掃描輸出第二列低電平
key_dr1=1;
key_dr2=0;
key_dr3=1;
key_dr4=1;

uiKeyTimeCnt=0;//延時計數(shù)器清零
ucKeyStep++; //切換到下一個運(yùn)行步驟
break;

case 5: //此處的小延時用來等待剛才列輸出信號穩(wěn)定,再判斷輸入信號。不是去抖動延時。
uiKeyTimeCnt++;
if(uiKeyTimeCnt>1)
{
uiKeyTimeCnt=0;
ucKeyStep++; //切換到下一個運(yùn)行步驟
}
break;

case 6:
if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
ucKeyStep++;//如果沒有按鍵按下,切換到下一個運(yùn)行步驟
ucKeyLock=0;//按鍵自鎖標(biāo)志清零
uiKeyTimeCnt=0; //按鍵去抖動延時計數(shù)器清零,此行非常巧妙

}
else if(ucKeyLock==0)//有按鍵按下,且是第一次觸發(fā)
{
if(key_sr1==0&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=2;//觸發(fā)2號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S2鍵
}

}
else if(key_sr1==1&&key_sr2==0&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=6;//觸發(fā)6號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S6鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==0&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=10;//觸發(fā)10號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S9鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==0)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=14;//觸發(fā)14號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S13鍵
}

}

}
break;

case 7: //按鍵掃描輸出第三列低電平
key_dr1=1;
key_dr2=1;
key_dr3=0;
key_dr4=1;

uiKeyTimeCnt=0;//延時計數(shù)器清零
ucKeyStep++; //切換到下一個運(yùn)行步驟
break;

case 8: //此處的小延時用來等待剛才列輸出信號穩(wěn)定,再判斷輸入信號。不是去抖動延時。
uiKeyTimeCnt++;
if(uiKeyTimeCnt>1)
{
uiKeyTimeCnt=0;
ucKeyStep++; //切換到下一個運(yùn)行步驟
}
break;

case 9:
if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
ucKeyStep++;//如果沒有按鍵按下,切換到下一個運(yùn)行步驟
ucKeyLock=0;//按鍵自鎖標(biāo)志清零
uiKeyTimeCnt=0; //按鍵去抖動延時計數(shù)器清零,此行非常巧妙

}
else if(ucKeyLock==0)//有按鍵按下,且是第一次觸發(fā)
{
if(key_sr1==0&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=3;//觸發(fā)3號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S3鍵
}

}
else if(key_sr1==1&&key_sr2==0&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=7;//觸發(fā)7號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S7鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==0&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=11;//觸發(fā)11號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S11鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==0)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=15;//觸發(fā)15號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S15鍵
}

}

}
break;

case 10: //按鍵掃描輸出第四列低電平
key_dr1=1;
key_dr2=1;
key_dr3=1;
key_dr4=0;

uiKeyTimeCnt=0;//延時計數(shù)器清零
ucKeyStep++; //切換到下一個運(yùn)行步驟
break;

case 11: //此處的小延時用來等待剛才列輸出信號穩(wěn)定,再判斷輸入信號。不是去抖動延時。
uiKeyTimeCnt++;
if(uiKeyTimeCnt>1)
{
uiKeyTimeCnt=0;
ucKeyStep++; //切換到下一個運(yùn)行步驟
}
break;

case 12:
if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
ucKeyStep=1;//如果沒有按鍵按下,返回到第一步,重新開始掃描
ucKeyLock=0;//按鍵自鎖標(biāo)志清零
uiKeyTimeCnt=0; //按鍵去抖動延時計數(shù)器清零,此行非常巧妙

}
else if(ucKeyLock==0)//有按鍵按下,且是第一次觸發(fā)
{
if(key_sr1==0&&key_sr2==1&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=4;//觸發(fā)4號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S4鍵
}

}
else if(key_sr1==1&&key_sr2==0&&key_sr3==1&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=8;//觸發(fā)8號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S8鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==0&&key_sr4==1)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=12;//觸發(fā)12號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S12鍵
}

}
else if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==0)
{
uiKeyTimeCnt++;//去抖動延時計數(shù)器
if(uiKeyTimeCnt>const_key_time)
{
uiKeyTimeCnt=0;
ucKeyLock=1;//自鎖按鍵置位,避免一直觸發(fā),只有松開按鍵,此標(biāo)志位才會被清零
ucKeySec=16;//觸發(fā)16號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S16鍵
}

}

}
break;


}


}


void key_service() //第三區(qū) 按鍵服務(wù)的應(yīng)用程序
{
switch(ucKeySec) //按鍵服務(wù)狀態(tài)切換
{
case 1:// 1號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S1鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 2:// 2號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S2鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 3:// 3號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S3鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 4:// 4號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S4鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 5:// 5號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S5鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 6:// 6號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S6鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 7:// 7號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S7鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 8:// 8號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S8鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 9:// 9號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S9鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 10:// 10號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S10鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 11:// 11號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S11鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 12:// 12號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S12鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 13:// 13號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S13鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 14:// 14號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S14鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 15:// 15號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S15鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
case 16:// 16號鍵 對應(yīng)朱兆祺學(xué)習(xí)板的S16鍵

uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
ucKeySec=0;//響應(yīng)按鍵服務(wù)處理程序后,按鍵編號清零,避免一致觸發(fā)
break;
}
}



void T0_time() interrupt 1
{
TF0=0;//清除中斷標(biāo)志
TR0=0; //關(guān)中斷

key_scan(); //按鍵掃描函數(shù)

if(uiVoiceCnt!=0)
{
uiVoiceCnt--; //每次進(jìn)入定時中斷都自減1,直到等于零為止。才停止鳴叫
beep_dr=0;//蜂鳴器是PNP三極管控制,低電平就開始鳴叫。
}
else
{
; //此處多加一個空指令,想維持跟if括號語句的數(shù)量對稱,都是兩條指令。不加也可以。
beep_dr=1;//蜂鳴器是PNP三極管控制,高電平就停止鳴叫。
}


TH0=0xf8; //重裝初始值(65535-2000)=63535=0xf82f
TL0=0x2f;
TR0=1;//開中斷
}


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ù)量
{
; //一個分號相當(dāng)于執(zhí)行一條空語句
}
}
}


void initial_myself()//第一區(qū) 初始化單片機(jī)
{


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é)中,有的人咋看我的按鍵掃描代碼,會覺得代碼太多了。我一直認(rèn)為,只要單片機(jī)容量夠,代碼多一點(diǎn)少一點(diǎn)并不重要,只要不影響運(yùn)行效率就行。而且有時候,代碼寫多一點(diǎn),可讀性非常強(qiáng),修改起來也非常方便。如果一味的追求壓縮代碼,就會刻意用很多循環(huán),數(shù)組等元素,代碼雖然緊湊了,但是可分離性,可改性,可讀性就沒那么強(qiáng)。我說那么多并不是因?yàn)槲壹夹g(shù)有限而不懂壓縮,就找個借口敷衍大家,不信?我下一節(jié)把這節(jié)的代碼壓縮一下分享給大家。凡是相似度高的那部分代碼都可以壓縮,具體怎么壓縮?欲知詳情,請聽下回分解-----發(fā)的壓縮代碼編程。


關(guān)鍵詞: 矩陣鍵盤單個觸

評論


技術(shù)專區(qū)

關(guān)閉