简易电子琴

2023-12-18 20:56:47

#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; ? ?//定义变量储存按键值

sbit sound=P3^7; ? ?//将sound位定义为P3.7
unsigned int C; ? ? //全局变量,储存定时器的定时常数
unsigned int f; ? ? //全局变量,储存音阶的频率

//以下是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=(46083/f)*10;?? ? ? //计算定时常数
? TH0=(8192-C)/32; ? ? //可证明这是13位计数器TH0高8位的赋初值方法
? TL0=(8192-C)%32; ? ? //可证明这是13位计数器TL0低5位的赋初值方法
? TR0=1; ? ? ? ? ? ? ? //开定时T0
? delay(); ? ? ? ? ? ? //延时200ms,播放音频
? TR0=0; ? ? ? ? ? ? ? //关闭定时器
? sound=1; ? ? ? ?//关闭蜂鸣器
? keyval=0xff;?? ? ? ?//播放按键音频后,将按键值更改,停止播放
}
/*******************************************
函数功能:主函数
******************************************/?? ?
void main(void)
? { ?
? ?? ? ? 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位赋初值?? ? ?? ??? ??? ??? ??? ??? ??? ? ??
?? ??? ??? ?
?? ??? ??? ??? ? while(1) ? ? ? ? ? ?//无限循环
?? ??? ??? ??? ? ? {
?? ??? ??? ??? ? ? ?switch(keyval)
?? ??? ??? ??? ??? ? ? ?{
?? ??? ??? ??? ??? ??? ? ? ?case 1:f=dao; ? ? ? ? ? ?//如果第1个键按下,将中音1的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ??? ? case 2:f=l_xi; ? ? ? ? ? //如果第2个键按下,将低音7的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
? ? ? ? ? ? ? ? ? ? ? case 3:f=l_la; ? ? ? ? ? //如果第3个键按下,将低音6的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ? ? ?case 4:f=l_sao; ? ? ? ? ?//如果第4个键按下,将低音5的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ??? ? case 5:f=sao; ? ? ? ? ? ? //如果第5个键按下,将中音5的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ??? ? case 6:f=fa; ? ? ? ? ? ? ?//如果第6个键按下,将中音4的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ? Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
? ? ? ? ? ? ? ? ? ? ? case 7:f=mi; ? ? ? ? ? ? //如果第7个键按下,将中音3的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;?
?? ??? ??? ??? ??? ??? ? ? ?case 8:f=re; ? ? ? ? ? ? ?//如果第8个键按下,将中音2的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ??? ? case 9:f=h_re; ? ? ? ? ? //如果第9个键按下,将高音2的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ? Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ??? ? case 10:f=h_dao; ? ? ? ? ? //如果第10个键按下,将高音1的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
? ? ? ? ? ? ? ? ? ? ? case 11:f=xi; ? ? ? ? ? ?//如果第11个键按下,将中音7的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ? ? ?case 12:f=la; ? ? ? ? ? ?//如果第12个键按下,将中音6的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;?
?? ??? ??? ??? ??? ??? ??? ? case 13:f=h_la; ? ? ? ? ?//如果第13个键按下,将高音6的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ??? ? case 14:f=h_sao; ? ? ? ? ?//如果第14个键按下,将高音5的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
? ? ? ? ? ? ? ? ? ? ? case 15:f=h_fa; ? ? ? ? ?//如果第15个键按下,将高音4的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break;
?? ??? ??? ??? ??? ??? ? ? ?case 16:f=h_mi; ? ? ? ? ? //如果第16个键按下,将高音3的频率赋给f
?? ??? ??? ??? ??? ??? ??? ? ? ?Output_Sound(); ? ?//转去计算定时常数?
?? ??? ??? ??? ??? ??? ??? ??? ? break; ?
?? ??? ? ? ? ? ? ? }?
?? ??? ? ??? ?}?? ?
?? ?}?
/**************************************************************
函数功能:定时器T0的中断服务子程序,使P3.7引脚输出音频方波
**************************************************************/?

? 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
? ? ?P1=0xf0; ? ? ? ? ? ? ? ? //所有行线置为低电平“0”,所有列线置为高电平“1”
?? ? ?if((P1&0xf0)!=0xf0) ? ? ?//列线中有一位为低电平“0”,说明有键按下
?? ? ? {
?? ??? ??? ? ? ? ? ? delay20ms(); ? ? ? ? ? //延时一段时间、软件消抖
?? ??? ??? ??? ? ?if((P1&0xf0)!=0xf0) ? ? ?//确实有键按下
?? ??? ??? ??? ? ? ?{
?? ??? ??? ??? ? ? ? ? P1=0xfe; ? ? ? ? ? ? //第一行置为低电平“0”(P1.0输出低电平“0”)
?? ??? ??? ??? ? ? ? ? if(P14==0) ? ? ? ? ? //如果检测到接P1.4引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=1; ? ? ? ? ? ?//可判断是S1键被按下
?? ??? ??? ? ? ? ? ? ? if(P15==0) ? ? ? ? ? ? //如果检测到接P1.5引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=2; ? ? ? ? ? ?//可判断是S2键被按下
?? ??? ??? ? ? ? ? ? ? if(P16==0) ? ? ? ? ? ? //如果检测到接P1.6引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=3; ? ? ? ? ? ?//可判断是S3键被按下
?? ??? ??? ? ? ? ? ? ? if(P17==0) ? ? ? ? ? ?//如果检测到接P1.7引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=4; ? ? ? ? ? //可判断是S4键被按下
?? ??? ??? ?
?? ??? ??? ? ? ? ? ? ? P1=0xfd; ? ? ? ? ? ? //第二行置为低电平“0”(P1.1输出低电平“0”)
?? ??? ??? ??? ? ? ? ?if(P14==0) ? ? ? ? ? //如果检测到接P1.4引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=5; ? ? ? ? ? ?//可判断是S5键被按下
?? ??? ??? ? ? ? ? ? ? if(P15==0) ? ? ? ? ? ? //如果检测到接P1.5引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=6; ? ? ? ? ? ?//可判断是S6键被按下
?? ??? ??? ? ? ? ? ? ? if(P16==0) ? ? ? ? ? ? //如果检测到接P1.6引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=7; ? ? ? ? ? ?//可判断是S7键被按下
?? ??? ??? ? ? ? ? ? ? if(P17==0) ? ? ? ? ? ?//如果检测到接P1.7引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=8; ? ? ? ? ? //可判断是S8键被按下
?? ??? ??? ??? ??? ?
?? ??? ??? ? ? ? ? ? ? P1=0xfb; ? ? ? ? ? ? //第三行置为低电平“0”(P1.2输出低电平“0”)
?? ??? ??? ??? ? ? ?if(P14==0) ? ? ? ? ?//如果检测到接P1.4引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=9; ? ? ? ? ?//可判断是S9键被按下
?? ??? ??? ? ? ? ? ? ? if(P15==0) ? ? ? ? ? //如果检测到接P1.5引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=10; ? ? ? ? //可判断是S10键被按下
?? ??? ??? ? ? ? ? ? ? if(P16==0) ? ? ? ? ?//如果检测到接P1.6引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=11; ? ? ? ?//可判断是S11键被按下
?? ??? ??? ? ? ? ? ? ? if(P17==0) ? ? ? ? ?//如果检测到接P1.7引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=12; ? ? ? ?//可判断是S12键被按下
?? ??? ??? ?
?? ??? ??? ? ? ? ? ? ? P1=0xf7; ? ? ? ? ? ? //第四行置为低电平“0”(P1.3输出低电平“0”)
?? ??? ??? ??? ? ? ?if(P14==0) ? ? ? ? ?//如果检测到接P1.4引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=13; ? ? ? ? ?//可判断是S13键被按下
?? ??? ??? ? ? ? ? ? ? if(P15==0) ? ? ? ? ? //如果检测到接P1.5引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=14; ? ? ? ? //可判断是S14键被按下
?? ??? ??? ? ? ? ? ? ? if(P16==0) ? ? ? ? ?//如果检测到接P1.6引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=15; ? ? ? ?//可判断是S15键被按下
?? ??? ??? ? ? ? ? ? ? if(P17==0) ? ? ? ? ?//如果检测到接P1.7引脚的列线为低电平“0”
?? ??? ??? ? ? ? ? ? ? ? keyval=16; ? ? ? ?//可判断是S16键被按下
?? ??? ??? ? ? ? }
?? ??? ?}
? ? ?TR1=1; ? ? ? ? ? ? ? ? ? ?//开启定时器T1
? ? ?TH1=(65536-500)/256; ?//定时器T1的高8位赋初值
?? ? ?TL1=(65536-500)%256; ?//定时器T1的高8位赋初值?? ??? ??
?}

?? ??? ?
?

文章来源:https://blog.csdn.net/qq_46065313/article/details/135069883
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。