#include<reg51.h> //包含51單片機寄存器定義的頭文件
sbit P14=P1^4; //將P14位定義為P1.4引腳
sbit P15=P1^5; //將P15位定義為P1.5引腳
sbit P16=P1^6; //將P16位定義為P1.6引腳
sbit P17=P1^7; //將P17位定義為P1.7引腳
unsigned char keyval; //定義變量儲存按鍵值
unsigned char q; //定義變量儲存按鍵值
sbit sound=P2^0; //將sound位定義為P2.0
sbit P22=P2^2; //將P22位定義為P2.2引腳
unsigned int C; //全局變量,儲存定時器的定時常數
unsigned int f; //全局變量,儲存音階的頻率
unsigned char s; //全局變量,儲存節拍
//定義 KEY_PORT 為 P1 端口
#define KEY_PORT P1
//以下是C調低音的音頻宏定義
#define l_dao 262 //將“l_dao”宏定義為低音“1”的頻率262Hz
#define l_re 286 //將“l_re”宏定義為低音“2”的頻率286Hz
#define l_mi 311 //將“l_mi”宏定義為低音“3”的頻率311Hz
#define l_fa 349 //將“l_fa”宏定義為低音“4”的頻率349Hz
#define l_sao 392 //將“l_sao”宏定義為低音“5”的頻率392Hz
#define l_la 440 //將“l_a”宏定義為低音“6”的頻率440Hz
#define l_xi 494 //將“l_xi”宏定義為低音“7”的頻率494Hz
//以下是C調中音的音頻宏定義
#define dao 523 //將“dao”宏定義為中音“1”的頻率523Hz
#define re 587 //將“re”宏定義為中音“2”的頻率587Hz
#define mi 659 //將“mi”宏定義為中音“3”的頻率659Hz
#define fa 698 //將“fa”宏定義為中音“4”的頻率698Hz
#define sao 784 //將“sao”宏定義為中音“5”的頻率784Hz
#define la 880 //將“la”宏定義為中音“6”的頻率880Hz
#define xi 987 //將“xi”宏定義為中音“7”的頻率53
//以下是C調高音的音頻宏定義
#define h_dao 1046 //將“h_dao”宏定義為高音“1”的頻率1046Hz
#define h_re 1174 //將“h_re”宏定義為高音“2”的頻率1174Hz
#define h_mi 1318 //將“h_mi”宏定義為高音“3”的頻率1318Hz
#define h_fa 1396 //將“h_fa”宏定義為高音“4”的頻率1396Hz
#define h_sao 1567 //將“h_sao”宏定義為高音“5”的頻率1567Hz
#define h_la 1760 //將“h_la”宏定義為高音“6”的頻率1760Hz
#define h_xi 1975 //將“h_xi”宏定義為高音“7”的頻率1975Hz
/**************************************************************
函數功能:軟件延時子程序
**************************************************************/
void delay20ms(void)
{
unsigned char i,j;
for(i=0;i<100;i++)
for(j=0;j<60;j++)
;
}
/*******************************************
函數功能:節拍的延時的基本單位,延時200ms
******************************************/
void delay()
{
unsigned char i,j;
for(i=0;i<250;i++)
for(j=0;j<250;j++);
}
/*******************************************
函數功能:輸出音頻
入口參數:f
******************************************/
void Output_Sound(void)
{
C=(50000/f)*10; //計算定時常數
TH0=(8192-C)/32; //可證明這是13位計數器TH0高8位的賦初值方法
TL0=(8192-C)%32; //可證明這是13位計數器TL0低5位的賦初值方法
TR0=1; //開定時T0
for(s=0;s<2;s++)
delay(); //延時200ms,播放音頻
TR0=0; //關閉定時器
sound=1; //關閉蜂鳴器
keyval=0xff; //播放按鍵音頻后,將按鍵值更改,停止播放
}
/*******************************************
函數功能:按鍵檢測
輸出參數:keyval
******************************************/
unsigned char KeyScan(void)
{
KEY_PORT = 0x0f; // P1.0-1.3輸出高電平,P1.4-P1.7輸出低電平
if (KEY_PORT != 0x0f) // 檢測是否有鍵按下
{
delay20ms(); // 消抖
if (KEY_PORT != 0x0f) // 確認按下
{
// 行
switch (KEY_PORT)
{
case 0x07 : keyval = 1; break;
case 0x0b : keyval = 2; break;
case 0x0d : keyval = 3; break;
case 0x0e : keyval = 4; break;
default : break;
}
// 列
KEY_PORT = 0xf0;
switch (KEY_PORT)
{
case 0x70: keyval = keyval + 0; break;
case 0xb0: keyval = keyval + 4; break;
case 0xd0: keyval = keyval + 8; break;
case 0xe0: keyval = keyval + 12; break;
}
return keyval;
}
}
return 0;
}
/*******************************************
函數功能:主函數
******************************************/
void main(void)
{
unsigned char n=0,m=0,k=0,z,i=0;
unsigned char pu[92]; //定義數組pu[]用于存儲鍵盤掃描結果,最多儲存92個按鍵
while(1)
{
EA=1; //開總中斷
ET0=1; //定時器T0中斷允許
ET1=1; //定時器T1中斷允許
TR1=1; //定時器T1啟動,開始鍵盤掃描
TMOD=0x10; //分別使用定時器T1的模式1,T0的模式0
TH1=(65536-500)/256; //定時器T1的高8位賦初值
TL1=(65536-500)%256; //定時器T1的高8位賦初值
q=KeyScan();
while(n!=1&&m<=i) //無限循環
{
if(P22==0) //當P22==0,播放之前的按鍵
i=m-1;
else{i=m;}
switch(q)
{
case 1:f=dao; //如果第1個鍵按下,將中音1的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=dao; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 2:f=l_xi; //如果第2個鍵按下,將低音7的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=l_xi; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 3:f=l_la; //如果第3個鍵按下,將低音6的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=l_la; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 4:f=l_sao; //如果第4個鍵按下,將低音5的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=l_sao; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 5:f=sao; //如果第5個鍵按下,將中音5的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=sao; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 6:f=fa; //如果第6個鍵按下,將中音4的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=fa; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 7:f=mi; //如果第7個鍵按下,將中音3的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=mi; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 8:f=re; //如果第8個鍵按下,將中音2的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=re; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 9:f=h_re; //如果第9個鍵按下,將高音2的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=h_re; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 10:f=h_dao; //如果第10個鍵按下,將高音1的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=h_dao; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 11:f=xi; //如果第11個鍵按下,將中音7的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=xi; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 12:f=la; //如果第12個鍵按下,將中音6的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=la; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 13:f=h_la; //如果第13個鍵按下,將高音6的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=h_la; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 14:f=h_sao; //如果第14個鍵按下,將高音5的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=h_sao; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 15:f=h_fa; //如果第15個鍵按下,將高音4的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=h_fa; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 16:f=h_mi; //如果第16個鍵按下,將高音3的頻率賦給f
Output_Sound(); //轉去計算定時常數
pu[m]=h_mi; //將鍵盤掃描的結果付數組pu[]的元素
m++; i++; //付值完畢m自動加1用于對數組的下一個元素付值
break;
case 17:n=1; //n==1,跳出循環
break;
}
}
while(pu[k]!=0) //播放通過按鍵寫入的音樂
{
f=pu[k];
Output_Sound();
k++;
}
for(z=0;z<41;z++) //清空pu[]
{
pu[z]=0;
}
}
}
/**************************************************************
函數功能:定時器T0的中斷服務子程序,使P2.0引腳輸出音頻方波
**************************************************************/
void Time0_serve(void ) interrupt 1 using 1
{
TH0=(8192-C)/32; //可證明這是13位計數器TH0高8位的賦初值方法
TL0=(8192-C)%32; //可證明這是13位計數器TL0低5位的賦初值方法
sound=!sound; //將P3.7引腳取反,輸出音頻方波
}
/**************************************************************
函數功能:定時器T1的中斷服務子程序,進行鍵盤掃描,判斷鍵位
**************************************************************/
void time1_serve(void) interrupt 3 using 2 //定時器T1的中斷編號為3,使用第2組寄存器
{
TR1=0; //關閉定時器T0
q=KeyScan();
TR1=1; //開啟定時器T1
TH1=(65536-500)/256; //定時器T1的高8位賦初值
TL1=(65536-500)%256; //定時器T1的高8位賦初值
}
|