基于51單片機(jī)的4×4矩陣鍵盤電子琴
前些日子,做而論道寫了一篇關(guān)于單片機(jī)發(fā)音的文章,后來,就不斷有網(wǎng)友來電詢問單片機(jī)電子琴的設(shè)計方法。
以前制作過一個24鍵(獨(dú)立按鍵)的,程序是用匯編語言寫的,估計多數(shù)人看不了。
下面,把新設(shè)計的16按鍵的電子琴,公布給網(wǎng)友。
電路圖如下:
制作說明:
單片機(jī)采用51系列的都行,AT89C2051也可;
圖中沒有畫出復(fù)位和晶振電路,實(shí)際制作時,不可省略,晶振可以使用11.0592或12MHz;
揚(yáng)聲器應(yīng)該按照圖中給出的附圖加上驅(qū)動電路;
顯示器及七段譯碼器不接,單片機(jī)電子琴也可以正常工作。
74LS47和數(shù)碼管之間,應(yīng)該接上“限流電阻”,約470歐姆即可。
C語言程序如下:
/*************************************************************
*程序功能:對4×4矩陣鍵盤進(jìn)行掃描,顯示鍵值和輸出音響
**************************************************************/
#include
#include
#defineuintunsignedint
#defineucharunsignedchar
sbitSPK=P3^7;//P3.7外接揚(yáng)聲器
uintFreqTemp;
unsignedintcodeFreqtab[]={//定時半周期的初始值
64021,64103,64260,64400,//低音3456
64524,64580,64684,64777,//低音7,中音123
64820,64898,64968,65030,//中音4567
65058,65110,65157,65178};//高音1234
/*************************************************************
*函數(shù)功能:用掃描法讀P1外接4×4鍵盤
*函數(shù)返回:按下鍵:返回0~15、如無鍵按下:返回16
**************************************************************/
ucharKeyscan(void)
{
uchari,j,temp,Buffer[4]={0xfe,0xfd,0xfb,0xf7};
for(j=0;j<4;j++){//循環(huán)四次,掃描四行
P1=Buffer[j];//在低四位分別輸出一個低電平
_nop_();
temp=0x80;//計劃先讀出P1.7位
for(i=0;i<4;i++){//循環(huán)四次,檢查四列
if(!(P1&temp)){//從高四位,截取1位
return(i+j*4);//返回取得的按鍵值
}
temp>>=1;//換右邊一位
}}
return16;//沒有鍵按下就返回16
}
/**************************************************************
*函數(shù)功能:將參數(shù)分成十位、個位,分別顯示到P2
*輸入:k(鍵盤數(shù)值)
***************************************************************/
voidDisplay(uchark)
{
P2=((k/10)<<4)+(k%10);
}
/**************************************************************
*主函數(shù)
***************************************************************/
voidMain(void)
{
ucharKey_Value=16,Key_Temp1,Key_Temp2;//讀出的鍵值
TMOD=0x01;//T0定時方式1
ET0=1;//允許T0中斷
EX0=1;//允許X0中斷
EA=1;
while(1){
TR0=0;//暫不發(fā)音
Key_Temp1=Keyscan();//讀入按鍵
if(Key_Temp1!=16){//有鍵按下
Display(Key_Value);//顯示鍵值、延時消抖
Key_Temp2=Keyscan();//再讀一次
if(Key_Temp1==Key_Temp2){//兩次相等
Key_Value=Key_Temp1;//就確認(rèn)下來
FreqTemp=Freqtab[Key_Value];//根據(jù)鍵值,取出定時半周期的初始值
Display(Key_Value);//顯示
TR0=1;//啟動定時器,發(fā)音
while(Keyscan()<16);//等待釋放
SPK=1;//停止發(fā)音
}}}}
//===============================================
voidT0_INT(void)interrupt1
{
TL0=FreqTemp;//載入定時半周期的初始值
TH0=FreqTemp>>8;
SPK=~SPK;//發(fā)音
}
//===============================================