⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 lts.c

📁 用C语言编的带数码管显示的电子琴
💻 C
字号:
#include<reg52.h>
#define uint unsigned int
#define uchar unsigned char
#define ulong unsigned long 
sbit BEEP=P2^4;
sbit K1= P3^2;     

uchar keyscan(uchar key);//键盘输入函数
void delay(uint t);//延时函数
void scan(uint i);//扫描显示函数
void bi(ulong t);

uchar table[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E};
uchar th0_f; //在中断中装载的T0的值高8位 
uchar tl0_f; //在中断中装载的T0的值低8位 

/******T0的值,及输出频率对照表******/
uchar code freq[36*2]=
{ 
	0xA9,0xEF,//00220HZ ,1  //0 
	0x93,0xF0,//00233HZ ,1# 
	0x73,0xF1,//00247HZ ,2 
	0x49,0xF2,//00262HZ ,2# 
	0x07,0xF3,//00277HZ ,3 
	0xC8,0xF3,//00294HZ ,4 
	0x73,0xF4,//00311HZ ,4# 
	0x1E,0xF5,//00330HZ ,5  
	0xB6,0xF5,//00349HZ ,5# 
	0x4C,0xF6,//00370HZ ,6 
	0xD7,0xF6,//00392HZ ,6# 
	0x5A,0xF7,//00415HZ ,7     
	0xD8,0xF7,//00440HZ 1     //12 
	0x4D,0xF8,//00466HZ 1#    //13 
	0xBD,0xF8,//00494HZ 2     //14 
	0x24,0xF9,//00523HZ 2#    //15 
	0x87,0xF9,//00554HZ 3     //16 
	0xE4,0xF9,//00587HZ 4     //17 
	0x3D,0xFA,//00622HZ 4#    //18 
	0x90,0xFA,//00659HZ 5     //19 
	0xDE,0xFA,//00698HZ 5#    //20 
	0x29,0xFB,//00740HZ 6     //21 
	0x6F,0xFB,//00784HZ 6#    //22 
	0xB1,0xFB,//00831HZ 7     //23 
	0xEF,0xFB,//00880HZ `1 
	0x2A,0xFC,//00932HZ `1# 
	0x62,0xFC,//00988HZ `2 
	0x95,0xFC,//01046HZ `2# 
	0xC7,0xFC,//01109HZ `3 
	0xF6,0xFC,//01175HZ `4 
	0x22,0xFD,//01244HZ  `4# 
	0x4B,0xFD,//01318HZ `5 
	0x73,0xFD,//01397HZ `5# 
	0x98,0xFD,//01480HZ `6 
	0xBB,0xFD,//01568HZ `6# 
	0xDC,0xFD,//01661HZ `7    //35 
};

//****************************** 
//音乐符号串解释函数 
//入口:要解释的音乐符号串,输出的音调串,输出的时长串 
changedata(uchar *song,uchar *diao,uchar *jie) 
{ 
    uchar i,i1,j; 
    char gaodi; //高低+/-12音阶 
    uchar banyin;//有没有半个升音阶 
    uchar yinchang;//音长 
    uchar code jie7[8]={0,12,14,16,17,19,21,23}; //C调的7个值 
 
    *diao=*song; 
    for(i=0,i1=0;;) 
    { 
        gaodi=0; //高低=0 
        banyin=0;//半音=0 
        yinchang=4;//音长1拍 
        if((*(song+i)=='|') || (*(song+i)==' ')) i++;   //拍子间隔和一个空格过滤 
        switch(*(song+i))  
        {     
            case ',': gaodi=-12;i++;//低音 
            break; 
     
            case '`': gaodi=12;i++; //高音 
            break; 
        } 
 
        if(*(song+i)==0)  //遇到0结束 
        { 
            *(diao+i1)=0; //加入结束标志0 
            *(jie+i1)=0; 
            return; 
        } 
 
        j=*(song+i)-0x30; i++; //取出基准音 
        j=jie7[j]+gaodi; //加上高低音 
       
 
		yinc:   switch(*(song+i))  
        {     
            case '#': //有半音j加一个音阶  
            	i++;j++; 
    			goto yinc;  
             
 
            case '-': //有一个音节加长 
                yinchang+=4; 
                i++;  
                goto yinc; 
 
            case '_': //有一个音节缩短 
                yinchang/=2; 
                i++;  
    			goto yinc; 
             
             
            case '.': //有一个加半拍 
                yinchang=yinchang+yinchang/2;
    			i++; 
    			goto yinc; 
        } 
 
        *(diao+i1)=j; //记录音符 
        *(jie+i1)=yinchang; //记录音长 
        i1++;     
    } 
} 
//****************************************** 
//奏乐函数 
//入口:要演奏的音乐符号串 
void play(uchar *songdata) 
{ 
    uchar i,c,j=0; 
    uint  n; 
    uchar xdata diaodata[112]; //音调缓冲 
    uchar xdata jiedata[112];  //音长缓冲 
 
    changedata(songdata,diaodata,jiedata); //解释音乐符号串 
    TR0=1; 
    for(i=0;diaodata[i]!=0;i++)  //逐个符号演奏 
    { 
        tl0_f=freq[diaodata[i]*2]; //取出对应的定时值送给T0 
        th0_f=freq[diaodata[i]*2+1]; 
        for(c=0;c<jiedata[i];c++) //按照音长延时 
  		{ 
            for(n=0;n<32000;n++); 
   			if(K1==0)//发现按键,立即退出播放 
   			{  
   				delay(1);
				if(K1==0)
					{TR0=0;scan(0);} 
				return; 
   			} 
  		} 
        TR0=0; 
        for(n=0;n<500;n++);    //音符间延时 
   
        TR0=1; 
    } 
    TR0=0; 
} 
 
//仙剑 
uchar code xianjian[]={ 
"|3_3_3_2_3-|2_3_2_2_,6,6_,7_|12_1_,7,6_,5_|,6---|" 
"3_3_3_2_3.6_|5_6_5_5_22_3_|45_4_32_1_|3.--3_|" 
"67_6_55_3_|5--3_5_|26_5_32_3_|3---|" 
"26_6_6-|16_6_66_7_|`17_6_76_7_|3.--3_|" 
"67_6_55_3_|5--3_5_|67_6_76_7_|3---|" 
"26_6_6-|16_6_66_7_|`17_6_7.5_|6---|" 
}; 
uchar code song3[]={ 
"5-5_3_2_1_|3---|6-6_4_2_1_" 
",7--,5_|1.3_5.1_|,7.3_5 5_|" 
"6.7_`1.6_|6_5_5-3_2_|1.1_13_2_|" 
"1.1_12_3_|2.1_,62_3_|2-- ,5_|" 
"1.3_5.1_|,7.3_55_|6.7_`1.6_|" 
"6_5_5-3_2_|1.1_13_2_|1.1_12_3_" 
"2.,6_,71_2_|1--" 
}; 
//世上只有妈妈好 
uchar code mamahao[]={ 
"6.5_35|`16_5_6-|35_6_53_2_|1_,6_5_3_2-|" 
"2.3_55_6_|321-|5.3_2_1_,6_1_|,5--" 
}; 

//我是一只菠萝 
uchar code boluo[]={ 
"1-|2_4_3_2_55|5_6_3_4_22|2_4_3_2_1_`1_7_6_|5_4_2_3_11|" 
"2_4_3_2_55|5_6_3_4_22|2_4_3_2_1_5_2_3_|1---|" 
}; 
//小星星 
uchar code xingxing[]={ 
"1155|665-|4433|221-|" 
"5544|332-|5544|332-|" 
"1155|665-|4433|221-|" 
}; 
 
//老鼠爱大米 
uchar code dami[]={ 
"321-|3.2_1-|3231|3_6.5-|3_5_665|65-3_2_|122_3_2|12--|" 
}; 
 

/******主函数******/
void main(void) // 主程序 
{ 
	uchar k; 	
	uchar code jie8[8]={12,14,16,17,19,21,23,24};//1234567`1八个音符在频率表中的位置 
	TMOD = 0x01;  //使用定时器0的16位工作模式 
	TR0 = 0; 
	ET0 = 1; 
	EA = 1; 
 
	while(1)
	{
		k=keyscan();
		switch(k)
		{
			case 1:{scan(1);tl0_f=freq[jie8[0]*2];th0_f=freq[jie8[0]*2+1]; TR0 = 1;delay(400);BEEP=1;}break;
			case 2:{scan(2);tl0_f=freq[jie8[1]*2];th0_f=freq[jie8[1]*2+1]; TR0 = 1;delay(400);BEEP=1;}break;
			case 3:{scan(3);tl0_f=freq[jie8[2]*2];th0_f=freq[jie8[2]*2+1]; TR0 = 1;delay(400);BEEP=1;}break;
			case 4:{scan(4);tl0_f=freq[jie8[3]*2];th0_f=freq[jie8[3]*2+1]; TR0 = 1;delay(400);BEEP=1;}break;
			case 5:{scan(5);tl0_f=freq[jie8[4]*2];th0_f=freq[jie8[4]*2+1]; TR0 = 1;delay(500);BEEP=1;}break;
			case 6:{scan(6);tl0_f=freq[jie8[5]*2];th0_f=freq[jie8[5]*2+1]; TR0 = 1;delay(500);BEEP=1;}break;
			case 7:{scan(7);tl0_f=freq[jie8[6]*2];th0_f=freq[jie8[6]*2+1]; TR0 = 1;delay(500);BEEP=1;}break;
			case 8:scan(8);play(xianjian); break;
			case 9:scan(9);play(song3); break;
			case 10:scan(10);play(boluo); break;
			case 11:scan(11);play(xingxing); break;
			case 12:scan(12);play(mamahao); break;
			case 13:scan(13);play(dami); break;
		} 
		TR0 = 0;
	}
} 
/******键盘输入函数******/
uchar keyscan(uchar key)
{
   	uchar n;
   	P1=0xfe;
   	n=P1;
  	n&=0xf0;
  	if(n!=0xf0)
  	{
    	delay(20);
    	P1=0xfe;
		n=P1;
	 	n&=0xf0;
	 	if(n!=0xf0)
		{
	     	switch(n)
		 	{
		   		case(0xe0):key=12;break;
				case(0xd0):key=9;break;
				case(0xb0):key=5;break;
    			case(0x70):key=1;break;
    		}
   	 	}
   	}

   	P1=0xfd;
   	n=P1;
	n&=0xf0;
	if(n!=0xf0)
	{
		delay(20);
		P1=0xfd;
		n=P1;
		n&=0xf0;
		if(n!=0xf0)
		{
   			switch(n)
			{
    			case(0xe0):key=13;break;
    			case(0xd0):key=0;break;
	    		case(0xb0):key=6;break;
	    		case(0x70):key=2;break;
    		}	
  		}
	}

	P1=0xfb;
	n=P1;
	n&=0xf0;
	if(n!=0xf0)
	{
		delay(20);
		P1=0xfb;
		n=P1;
		n&=0xf0;
		if(n!=0xf0)
		{
   			switch(n)
			{
    			case(0xe0):key=14;break;
    			case(0xd0):key=10;break;
    			case(0xb0):key=7;break;
    			case(0x70):key=3;break;
    		}
   		}
	}

	P1=0xf7;
	n=P1;
	n&=0xf0;
	if(n!=0xf0)
	{
		delay(20);
		P1=0xf7;
		n=P1;
		n&=0xf0;
		if(n!=0xf0)
		{
	   		switch(n)
			{
	    		case(0xe0):key=15;break;
	    		case(0xd0):key=11;break;
	    		case(0xb0):key=8;break;
	    		case(0x70):key=4;break;
	    	}
	   }
	}	
	return (key);
}

/******延时函数******/
void delay(uint t)
{
	uint i,j;
	for(i=0;i<t;i++)
	{
		for(j=0;j<250;j++)
		{;}
	}
}

/******显示函数******/
void scan(uint i)
{
	
	P2=0xf0;
	P0=table[i];
}

/******峰鸣器函数******/
void bi(ulong t) 
{ 
	ulong c; 
	uint n; 
	for(c=0;c<t;c++) 
	{ 
		for(n=0;n<50;n++); //延时 
		BEEP=~BEEP;      //取反输出到喇叭的信号 
	} 
} 

//定时中断0,用于产生唱歌频率 
timer0() interrupt 1  
{ 
    TL0=tl0_f;TH0=th0_f;  //调入预定时值 
    BEEP=~BEEP;           //取反音乐输出IO 
} 


⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -