新聞中心

        EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > 單片機編程 音樂編程程序

        單片機編程 音樂編程程序

        作者: 時間:2016-11-17 來源:網(wǎng)絡 收藏
        #include

        #defineucharunsignedchar
        #defineuintunsignedint
        #defineulongunsignedlong

        本文引用地址:http://www.104case.com/article/201611/315417.htm


        sbitBEEP=P2^0;//喇叭輸出腳

        ucharth0_f;//在中斷中裝載的T0的值高8位
        uchartl0_f;//在中斷中裝載的T0的值低8位
        ucharkey;

        /*------------------------------------------------
        函數(shù)聲明
        ------------------------------------------------*/
        ucharkeyscan(void);//鍵盤掃描程序
        voiddelay(uinti);//延時子程序

        /*--------T0的值,及輸出頻率對照表--------------*/
        ucharcodefreq[36*2]={
        0xA9,0xEF,//0x3F,0XEE,//00220HZ,1//0
        0x93,0xF0,//0X3D,0XEF,//00233HZ,1#
        0x73,0xF1,//0X30,0XF0,//00247HZ,2
        0x49,0xF2,//0X18,0XF1,//00262HZ,2#
        0x07,0xF3,//0XE6,0XF1,//00277HZ,3
        0xC8,0xF3,//0XB7,0XF2,//00294HZ,4
        0x73,0xF4,//0X71,0XF3,//00311HZ,4#
        0x1E,0xF5,//0X2A,0XF4,//00330HZ,5
        0xB6,0xF5,//0XCF,0XF4,//00349HZ,5#
        0x4C,0xF6,//0X72,0XF5,//00370HZ,6
        0xD7,0xF6,//0X09,0XF6,//00392HZ,6#
        0x5A,0xF7,//0X97,0XF6,//00415HZ,7
        0xD8,0xF7,//0X20,0XF7,//00440HZ1//12
        0x4D,0xF8,//0X9F,0XF7,//00466HZ1#//13
        0xBD,0xF8,//0X18,0XF8,//00494HZ2//14
        0x24,0xF9,//0X88,0XF8,//00523HZ2#//15
        0x87,0xF9,//0XF3,0XF8,//00554HZ3//16
        0xE4,0xF9,//0X59,0XF9,//00587HZ4//17
        0x3D,0xFA,//0X05,0XFA,//00622HZ4#//18
        0x90,0xFA,//0X13,0XFA,//00659HZ5//19
        0xDE,0xFA,//0X66,0XFA,//00698HZ5#//20
        0x29,0xFB,//0XB9,0XFA,//00740HZ6//21
        0x6F,0xFB,//0X05,0XFB,//00784HZ6#//22
        0xB1,0xFB,//0X4D,0XFB,//00831HZ7//23
        0xEF,0xFB,//0X90,0XFB,//00880HZ`1
        0x2A,0xFC,//0XD0,0XFB,//00932HZ`1#
        0x62,0xFC,//0X0C,0XFC,//00988HZ`2
        0x95,0xFC,//0X44,0XFC,//01046HZ`2#
        0xC7,0xFC,//0X7B,0XFC,//01109HZ`3
        0xF6,0xFC,//0XAD,0XFC,//01175HZ`4
        0x22,0xFD,//0XDD,0XFC,//01244HZ`4#
        0x4B,0xFD,//0X0A,0XFD,//01318HZ`5
        0x73,0xFD,//0X35,0XFD,//01397HZ`5#
        0x98,0xFD,//0X5D,0XFD,//01480HZ`6
        0xBB,0xFD,//0X83,0XFD,//01568HZ`6#
        0xDC,0xFD,//0XA6,0XFD,//01661HZ`7//35
        };

        //定時中斷0,用于產(chǎn)生唱歌頻率
        timer0()interrupt1
        {
        TL0=tl0_f;TH0=th0_f;//調(diào)入預定時值
        BEEP=~BEEP;//取反音樂輸出IO
        }

        //音樂符號串解釋函數(shù)
        //入口:要解釋的音樂符號串,輸出的音調(diào)串,輸出的時長串
        changedata(uchar*song,uchar*diao,uchar*jie)
        {
        uchari,i1,j;
        chargaodi;//高低+/-12音階
        ucharbanyin;//有沒有半個升音階
        ucharyinchang;//音長
        ucharcodejie7[8]={0,12,14,16,17,19,21,23};//C調(diào)的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++;
        gotoyinc;
        case-://有一個音節(jié)加長
        yinchang+=4;
        i++;
        gotoyinc;
        case_://有一個音節(jié)縮短
        yinchang/=2;
        i++;
        gotoyinc;
        case.://有一個加半拍
        yinchang=yinchang+yinchang/2;
        i++;
        gotoyinc;
        }
        *(diao+i1)=j;//記錄音符
        *(jie+i1)=yinchang;//記錄音長
        i1++;
        }
        }

        //奏樂函數(shù)
        //入口:要演奏的音樂符號串
        voidplay(uchar*songdata)
        {
        uchari,c,j=0;
        uintn;
        uchardiaodata[48];//音調(diào)緩沖
        ucharjiedata[48];//音長緩沖
        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{key=keyscan();
        for(n=0;n<29500;n++);//29500
        if(key!=0xff)//((!K1)||(!K2)||(!K3)||(!K4))//發(fā)現(xiàn)按鍵,立即退出播放
        {
        TR0=0;
        return;
        }
        }
        TR0=0;
        for(n=0;n<460;n++);//460音符間延時
        TR0=1;
        }
        TR0=0;
        }

        //一分錢
        ucharcodeyifenqian[]={
        "5`1|6_`1_5|3_5_2_3_|5-|"
        "3_5_6_`1_|5_6_5_3_|1.3__|2-|"
        "3_2_1_2_|3-|6_5_3_5_|6-|"
        "5_`1_6_5_|3_5_2|5_2_3_2_|1-|"
        };

        //世上只有媽媽好
        ucharcodemamahao[]={
        "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--"
        };
        //找朋友
        ucharcodezhaopengyou[]={"5_6_5_6_|5_6_5|5_`1_7_6_|53|"
        "5_5_3_4_|5_5_3|1_4_3_2_|1_2_1|"
        };
        //茉莉花
        ucharcodemolihua[]={"33_5_6_`1_`1_6_|55_6_5-|33_5_6_`1_`1_6_|55_6_5-|"
        "5553_5_|665-|32_3_53_2_|11_2_1|"
        //"3_2_1_3_2.3_|56_`1_5-|23_5_2_3_1_,6_|,5-,61|"
        //"2.3_1_2-1_,6_|,5--"
        };
        //新年好
        ucharcodexinnianhao[]={
        "1_1_1,5|3_3_31|1_3_55|4_3_2-|"
        "2_3_4-|3_2_31|1_3_2,5|,7_2_1-|"
        };
        //小星星
        ucharcodexingxing[]={
        "1155|665-|4433|221-|"
        "5544|332-|5544|332-|"
        "1155|665-|4433|221-|"
        };
        //外婆的澎湖灣
        ucharcodewaipodephw[]={
        "3_5_5_5_6_`1_6_5_|`1_`1_`1_6_5-|"
        "`3_`3_`3_`3_`4_`3_`2_`1_|`2_`2_`2_`3_`2-|"
        "`3_`3_`3_`3_`4_`3_`2_`1_|6_`1_`1_6_5-|"
        };
        }

        //春天在哪里
        ucharcodechuntian[]={"3_3_3_1_|,5,5|3_3_3_1_|3-|5_5_3_1_|,5_,5_,5|,6_,7_1_3_|2-|"
        "3_3_3_1_|,5,5|3_3_3_1_|3-|5_6_5_6_|5_4_3_1_|,53|1-|"
        };

        //兩只老虎
        ucharcodelaohu[]={"1231|1231|345-|345-|5_6_5_4_31|5_6_5_4_31|"
        "151-|151-"
        };

        /*------------------------------------------------
        鍵盤掃描程序
        ------------------------------------------------*/
        ucharkeyscan(void)//鍵盤掃描函數(shù),使用行列反轉掃描法
        {
        ucharcord_h,cord_l;//行列值中間變量
        P3=0x0f;//行線輸出全為0
        cord_h=P3&0x0f;//讀入列線值
        if(cord_h!=0x0f)//先檢測有無按鍵按下
        {
        delay(100);//去抖
        if(cord_h!=0x0f)
        {
        cord_h=P3&0x0f;//讀入列線值
        P3=cord_h|0xf0;//輸出當前列線值
        cord_l=P3&0xf0;//讀入行線值
        return(cord_h+cord_l);//鍵盤最后組合碼值
        }
        }
        return(0xff);//返回該值
        }
        /*------------------------------------------------
        延時程序
        ------------------------------------------------*/
        voiddelay(uinti)//延時函數(shù)
        {
        while(i--);
        }
        //樂譜方式輸入的音樂播放,
        voidmain(void)//主程序
        {
        ucharcodejie8[8]={12,14,16,17,19,21,23,24};//1234567`1八個音符在頻率表中的位置
        uchari=0;
        ucharn=0;
        TMOD=0x01;//使用定時器0的16位工作模式
        TR0=0;
        ET0=1;
        EA=1;

        while(1)
        {
        key=keyscan();
        switch(key)
        {
        case0x7e:
        while(key==0x7e)
        {
        key=keyscan();
        }
        play(molihua);
        break;//0
        case0x7d:
        while(key==0x7d)
        {
        key=keyscan();
        }
        play(yifenqian);
        break;//1
        case0x7b:while(key==0x7b)
        {
        key=keyscan();
        }
        play(chuntian);
        break;//2
        case0x77:while(key==0x77)
        {
        key=keyscan();
        }
        play(waipodephw);
        break;//3
        case0xbe:while(key==0xbe)
        {
        key=keyscan();
        }
        play(xingxing);
        break;//4
        case0xbd:while(key==0xbd)
        {
        key=keyscan();
        }
        play(xinnianhao);
        break;//5
        case0xbb:while(key==0xbb)
        {
        key=keyscan();
        }
        play(mamahao);
        break;//6
        case0xb7:while(key==0xb7)
        {
        key=keyscan();
        }
        play(zhaopengyou);
        break;//7
        case0xde:while(!(key^0xde))
        {
        tl0_f=freq[jie8[0]*2];//置一個音符的值
        th0_f=freq[jie8[0]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }
        break;//8
        case0xdd:while(!(key^0xdd))
        {
        tl0_f=freq[jie8[1]*2];//置一個音符的值
        th0_f=freq[jie8[1]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }
        break;//9
        case0xdb:while(!(key^0xdb))
        {
        tl0_f=freq[jie8[2]*2];//置一個音符的值
        th0_f=freq[jie8[2]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }
        break;//a
        case0xd7:while(!(key^0xd7))
        {
        tl0_f=freq[jie8[3]*2];//置一個音符的值
        th0_f=freq[jie8[3]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }
        break;//b
        case0xee:while(!(key^0xee))
        {
        tl0_f=freq[jie8[4]*2];//置一個音符的值
        th0_f=freq[jie8[4]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }break;//c
        case0xed:while(!(key^0xed))
        {
        tl0_f=freq[jie8[5]*2];//置一個音符的值
        th0_f=freq[jie8[5]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }break;//d
        case0xeb:while(!(key^0xeb))
        {
        tl0_f=freq[jie8[6]*2];//置一個音符的值
        th0_f=freq[jie8[6]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }break;//e
        case0xe7:while(!(key^0xe7))
        {
        tl0_f=freq[jie8[7]*2];//置一個音符的值
        th0_f=freq[jie8[7]*2+1];
        TR0=1;
        key=keyscan();
        //for(n=0;n<10000;n++);//延時
        }break;//f
        default:break;
        }
        TR0=0;
        /*if(!K1)
        {
        tl0_f=freq[jie8[0]*2];//置一個音符的值
        th0_f=freq[jie8[0]*2+1];
        TR0=1;
        for(n=0;n<10000;n++);//延時
        }
        if(!K2)
        {
        tl0_f=freq[jie8[1]*2];//置一個音符的值
        th0_f=freq[jie8[1]*2+1];
        TR0=1;
        for(n=0;n<10000;n++);//延時
        }
        if(!K3)
        {
        tl0_f=freq[jie8[2]*2];//置一個音符的值
        th0_f=freq[jie8[2]*2+1];
        TR0=1;
        for(n=0;n<10000;n++);//延時
        }
        if(!K4)
        {
        tl0_f=freq[jie8[3]*2];//置一個音符的值
        th0_f=freq[jie8[3]*2+1];
        TR0=1;
        for(n=0;n<10000;n++);//延時
        }
        TR0=0;
        if(!K1)
        {
        while(!K1);
        play(molihua);//播放音樂
        }
        if(!K2)
        {
        while(!K2);
        play(dami);//播放音樂
        }
        if(!K3)
        {
        while(!K3);
        play(mamahao);//播放音樂
        }
        if(!K4)
        {
        switch(i)
        {
        case0:
        play(xianjian);//播放音樂
        break;
        case1:
        play(song3);//播放音樂
        break;
        case2:
        play(mamahao);//播放音樂
        break;
        case3:
        play(boluo);//播放音樂
        break;
        case4:
        play(xingxing);//播放音樂
        break;
        case5:
        play(dami);//播放音樂
        break;
        }
        i++;if(i==6)i=0;
        }*/
        }



        評論


        技術專區(qū)

        關閉
        主站蜘蛛池模板: 丹巴县| 南江县| 枣阳市| 五家渠市| 夏河县| 洪泽县| 清镇市| 武山县| 揭阳市| 五家渠市| 秭归县| 自贡市| 庆城县| 阿克苏市| 吐鲁番市| 克什克腾旗| 开封市| 梧州市| 池州市| 牡丹江市| 黑龙江省| 嘉定区| 高雄县| 墨玉县| 绵竹市| 肇庆市| 南皮县| 措美县| 闵行区| 镇安县| 桐乡市| 白河县| 乌拉特前旗| 镇坪县| 文安县| 宁津县| 仁寿县| 定州市| 定襄县| 保靖县| 格尔木市|