当前位置:首页 > 单片机 > 单片机
[导读]利用蜂鸣器和单片机演奏简单的音乐实验原理图 实验程序/* =========================================================== *//* ----------------------------------------------------------- *//* 曲谱存贮格式 uch

利用蜂鸣器单片机演奏简单的音乐

实验原理图

 

实验程序

/* =========================================================== */

/* ----------------------------------------------------------- */

/* 曲谱存贮格式 uchar code 数组名{音高,音长,音高,音长....} */

/* 音高由三位数字组成: */

/* 个位是表示 1~7 这七个音符. */

/* 十位是表示音符所在的音区,1-低音,2-中音,3-高音; */

/* 百位表示这个音符是否要升半音 0(不写)-不升,1-升半音。 */

/* 音长最多由三位数字组成: */

/* 个位表示音符的时值,其对应关系是: */

/* 数值(n) 0 1 2 3 4 5 6 */

/* --------------------------------------------- */

/* 几分音符 1 2 4 8 16 32 64 */

/* 即:音符=2^n ,这样做的目的是为了节省曲谱的存贮空间。 */

/* 十位表示音符的演奏效果(0-2),0-普通,1-连音,2-顿音。 */

/* 百位是符点位,0(不写)-无符点,1-有符点。 */

/* ----------------------------------------------------------- */

/* 调用演奏子程序的方法为: */

/* play(乐曲数组名,调号,升降八度,演奏速度,开始指针,结束指针) */

/* 调号(0-11)是指乐曲升多少个半音演奏;升降八度(1-3)是指在演奏 */

/* 在哪个八度演奏: 1-降八度,2-不升不降,3-升八度.开始指针(0- ) */

/* 是从哪个音符开始演奏,结束指针是演奏到哪个音符为止. */

/* ----------------------------------------------------------- */

//本程序用T0 来产生音调,用T1 产生音长

#include <reg51.h>

#define uchar unsigned char

#define yx 4/5 /* 定义普通音符演奏的长度分率 */

#define plen 2 /* 定义晶振的时钟周期(us) */

#define uchar unsigned char

#define uint unsigned int

sbit speaker=P3^5;

/* ------------------下面是曲谱 ------------------------------ */

uchar code sound[100]=

{25,2,23,3,25,3,31,1,26,2,31,3,26,3,25,1,25,2,21,3,22,3,23,2,22,3,21,3,22,0,

25,2,23,3,25,3,31,102,27,3,26,2,31,2,25,1,25,2,22,3,23,3,24,102,17,3, 21,0};

uchar tc0,tc1,sc0,sc1; /* 音长和音符两个计数器初值暂存 */

void play(sound,dh,sj,speed,point1,point2)

uchar code sound[]; /* 接受乐曲数组的地址 */

uchar speed,sj,dh; /* 速度、八度、调号 */

uint point1,point2; /* 乐曲开始、结束指针 */

{

uint code fftab[12]={262,277,294,311,330,349,369,392,415,440,466,494}; /* 频率表*/

uchar code stab[7]={0,2,4,5,7,9,11}; /* 1~7 在频率表中的位置 */

uchar code ltab[7]={1,2,4,8,16,32,64};

uchar tl,ts,sl,sm,sh,slen,xg,ii,fd;

uint point,hz,tc,sc,len,len0,len1,len2,len4,i,ftab[12];

speaker=1;

for(i=0;i<12;i++) /* 根据调号及升降八度来计算音符频率 */

{

ii=i+dh;

if(ii>11)

{

ii=ii-12;

ftab[i]=fftab[ii]*2;

}

else

ftab[i]=fftab[ii];

if(sj==1) ftab[i]>>=2;

if(sj==3) ftab[i]<<=2;

}

point=point1;

ts=sound[point];

tl=sound[point+1]; /* 读出第一个音符和它时时值 */

tc=65535-10000/plen; /* 算出10ms 的初装值 */

tc0=tc%256; /* 计算TL1 应装入的初值 */

tc1=tc/256; /* 计算TH1 应装入的初值 */

len0=12000/speed; /* 算出1 分音符的长度(几个10ms) */

len4=len0/4; /* 算出4 分音符的长度 */

len4=len4-len4*yx; /* 普通音最长间隔标准 */

TMOD=0x11;

TH1=tc1; TL1=tc0;

ET0=1; EA=1;

TR0=0; TR1=1;

while(point<=point2)

{

sl=ts%10; /* 计算出音符 */

sh=ts/100; /* 计算出是否升半 */

sm=ts/10%10; /* 计算出高低音 */

hz=ftab[stab[sl-1]+sh]; /* 查出对应音符的频率 */

if(sl!=0)

{

if (sm==1) hz>>=2; /* 若是低音 */

if (sm==3) hz<<=2; /* 若是高音 */

sc=(50000/hz)*10/plen; /* 计算脉冲个数 */

sc=65536-sc; /* 计算计数器初值 */

sc0=sc%256; /* 算出TL0 应装初值 */

sc1=sc/256; /* 算出TH0 应装初值 */

TH0=sc1; /* 装入初值 */

TL0=sc0+12; /* 加12 是对中断延时的补偿 */

}

slen=ltab[tl%10]; /* 算出是几分音符 */

xg=tl/10%10; /* 算出音符类型(0 普通1 连音2 顿音) */

fd=tl/100;

len=len0/slen; /* 算出连音音符演奏的长度(多少个10ms)*/

if (fd==1) len=len+len/2;

if(xg!=1)

if(xg==0) /* 算出普通音符的演奏长度 */

if (slen<=4)

len1=len-len4;

else

len1=len*yx;

else

len1=len/2; /* 算出顿音的演奏长度 */

else

len1=len;

if(sl==0) len1=0;

len2=len-len1; /* 算出不发音的长度 */

if (sl!=0)

{

TR0=1;

for(i=len1;i>0;i--) /* 发规定长度的音 */

{

while(TF1==0);

TH1=tc1; TL1=tc0;

TF1=0;

}

}

if(len2!=0)

{

TR0=0; speaker=1;

for(i=len2;i>0;i--) /* 音符间的间隔 */

{

while(TF1==0);

TH1=tc1; TL1=tc0;

TF1=0;

}

}

point+=2; /* 音符指针下移 */

ts=sound[point]; tl=sound[point+1]; /* 读出下一个音符和它时时值 */

}

}

void yin() interrupt 1 /* 音符发生程序(中断服务程序)*/

{

speaker=~speaker;

TH0=sc1; TL0=sc0;

}

//==============================================

main()

{

while(1)

{

play(sound,0,2,60,0,57);

play(sound,0,1,60,0,57);

play(sound,0,3,60,0,57);

play(sound,0,2,40,0,57);

play(sound,5,2,60,0,57);

play(sound,0,2,80,0,57);

}

}

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭