蜂鸣器的使用与简介(有源蜂鸣器和无源蜂鸣器)/基于STM32F103C8T6/用蜂鸣器来放歌/PWM波输出/定时器配置

在CSDN上面关于蜂鸣器的介绍并不少,在这里做一个总结,内容应该算是完整,还请耐心阅读。

1 有源蜂鸣器

有源蜂鸣器的使用很简单,因为已经内置震荡源,只要是通电就能发声。
这里是正点原子关于蜂鸣器的驱动电路图,用了一个S8050的三极管来驱动。
在这里插入图片描述
市面上的蜂鸣器模块一般是由S8550的三极管驱动的,这种模块蜂鸣器就实际来看是既能带动无源蜂鸣器也能够带动有源蜂鸣器
在这里插入图片描述
对于这种有源蜂鸣器的描述不过多赘述,只要引脚给低电平就行了。

2 无源蜂鸣器

发声机理

无源蜂鸣器的发声主要是靠引脚输出PWM波来控制,PWM波有两个很重要的地方:频率占空比
占空比相同的PWM波的频率不一定相同,占空比可以在宏观上决定给蜂鸣器电压的大小,来决定响度频率决定音调

音符与频率

对于乐理,如果有什么不对的地方还请斧正。
对于音符,有对应的频率对照表C调音符与频率对照表
在这里插入图片描述

同样的频率,在不同的音调里面有不同的对应频率,比如D调的低音1 DO在C调里面是2 RE,频率都是294Hz,如图所示,更多的对照表都在上面的链接中了。之所以说明这个,是想提个醒,如果想让曲子的音调更准,需要对每个曲子的中低高音进行曲调的适配。具体怎么做,请继续往下看。
在这里插入图片描述
代码部分
首先在.h文件用define的方式定义各个音名,低音为L,中音为M,高音为H,并把他们的频率值赋给音名。

//定义低音音名 (单位:HZ)
#define L1 262
#define L2 294
#define L3 330
#define L4 349
#define L5 392
#define L6 440
#define L7 494

//定义中音音名
#define M1 523
#define M2 587
#define M3 659
#define M4 698
#define M5 784
#define M6 880
#define M7 988

//定义高音音名
#define H1 1047
#define H2 1175
#define H3 1319
#define H4 1397
#define H5 1568
#define H6 1760
#define H7 1976

然后,在.h文件中定义演奏速度和结构体乐谱。

//全音符所占的时值,单位ms,决定乐谱演奏速度
#define T 2200

typedef struct
{
	short mName;//音名:取值L1~L7、M1~M7、H1~H7分别表示低音、中音、高音的1234567,取0表示休止符
	short mTime;//时值:取值T、T/2、T/4、T/8、T/16、T/32分别表示全音符、二分音符、四音符、八音符...取0表示演奏结束
	
}tNote;

时值的计算

BPM是Beat Per Minute的简称,中文名为拍子数,释义为每分钟节拍数的单位。最浅显的概念就是在一分钟的时间段落之间,所发出的声音节拍的数量,这个数量的单位便是BPM。这个与上面的定义的T有关。有的曲子会在谱子前面表明BPM,一般是100个全音符每分钟。
在这里插入图片描述左上角代表该曲子是C大调,四四拍节奏,一四分音符为一拍,四拍为一小节。按照四分音符为一拍,可以有以下对照表。

音符名称写法时值
全音符4­­­ – – –四拍
二分音符4 –二拍
四分音符4一拍
八分音符4半拍
十六分音符以此类推四分之一拍

我们可以确定每一个音符的时值,在这里插入图片描述是一个低音6,八分音符,占半拍,曲子是四四拍的曲子,值为T/8。
这里不给再多的例子了,谱曲给到大家。

const tNote MyScore[]=
{
  {L6,T/8},{M3,T/8},{M3,T/8},{M3,T/8},{M3,T/4},{M3,T/8},{M2,T/8},
  {M1,T/8},{M1,T/16},{M2,T/16},{M1,T/8},{L7,T/8},{L6,T/4},
  {M6,T/8},{M6,T/8},{M6,T/8},{M6,T/8},{M6,T/4},{M6,T/8},{M5,T/8},
    
  {M3,T/8},{M5,T/8},{M5,T/8},{M4,T/8},{M3,T/2},
  {M3,T/8},{M6,T/8},{M6,T/8},{M5,T/8},{M3,T/4},{M3,T/8},{M2,T/8},
  {M1,T/8},{M2,T/8},{M1,T/8},{L7,T/8},{L6,T/4},{L3,T/4},
  
  {L3,T/8},{M1,T/8},{M1,T/8},{L7,T/8},{L6,T/4},{L6,T/8},{M3,T/8},
  {M2,T/8},{M2,T/16},{M1,T/16},{L7,T/8},{L5,T/8},{L6,T/2},
  ///
  {L6,T/8},{M3,T/8},{M3,T/8},{M3,T/8},{M3,T/4},{M3,T/8},{M2,T/8},
  {M1,T/8},{M1,T/16},{M2,T/16},{M1,T/8},{L7,T/8},{L6,T/4},
  {M6,T/8},{M6,T/8},{M6,T/8},{M6,T/8},{M6,T/4},{M6,T/8},{M5,T/8},
    
  {M3,T/8},{M5,T/8},{M5,T/8},{M4,T/8},{M3,T/2},
  {M3,T/8},{M6,T/8},{M6,T/8},{M5,T/8},{M3,T/4},{M3,T/8},{M2,T/8},
  {M1,T/8},{M2,T/8},{M1,T/8},{L7,T/8},{L6,T/4},{L3,T/4},
  
  {L3,T/8},{M1,T/8},{M1,T/8},{L7,T/8},{L6,T/4},{L6,T/8},{M3,T/8},
  {M2,T/8},{M2,T/16},{M1,T/16},{L7,T/8},{L5,T/8},{L6,T/2},

  {0,0}//结束
};

代码部分

首先,是初始化STM32的定时器,其实也就是初始化PWM波输出,这里以定时器4的通道3来举例,也就是引脚PB8(STM32F103C8T6)

void TIM4_PWM_Init(u16 arr,u16 psc)
{  
	GPIO_InitTypeDef GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	TIM_OCInitTypeDef  TIM_OCInitStructure;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//B的时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);//定时器时钟	
 
   //设置该引脚为复用输出功能,输出TIM4 CH3 PWM脉冲波形
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //复用推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIO
 
   //初始化TIM4
	TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
	TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值 
	TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
	TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
	
	//初始化TIM4 Channe3 PWM模式	 
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //选择定时器模式:TIM脉冲宽度调制模式1
 	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //输出极性:TIM输出比较极性高
	TIM_OCInitStructure.TIM_Pulse = 0;
	TIM_OC3Init(TIM4, &TIM_OCInitStructure);  //根据T指定的参数初始化外设
 
	TIM_OC3PreloadConfig(TIM4, TIM_OCPreload_Enable); //使能TIM4在CCR3上的预装载寄存器
	
	TIM_CtrlPWMOutputs(TIM4,ENABLE);	//MOE 主输出使能		
 
	TIM_Cmd(TIM4, ENABLE);  //使能TIM4

}

//主函数初始化示例:
TIM4_PWM_Init(14399,719);//100kHz计数频率

这里对代码进行简略的讲解:
1.
首先是频率设置,这里用到了TIM4_PWM_Init(14399,719); 14399这个数值本身并不重要,它可以是任何的数值,因为想让蜂鸣器发出不同的音调,你就需要改变PWM输出的频率,所以,前面装填的arr的值是会随时更改的,每变一次音调就要更改一次频率。这里给出频率计算公式:

Tout= ((arr+1)*(psc+1))/Tclk;

这个严格来说是定时器中断频率的计算公式,但实际上,即使是在PWM输出的情况下进行定时器中断的配置也是可以的,他们两个是不干扰的,可以独立运行,但是若这样设置,则PWM波的输出频率和定时器中断频率是一致的。
psc是进行分频的参数,也就是定时器的时钟除以psc+1是计数器计数的频率,以设置的计数频率计数到arr+1为一个周期,这个周期的倒数,便是最后所输出的频率,这便是定时器的工作原理。
根据上述公式进行计算,72000k/720 = 100k,也就是计数器频率为100k,这个数值将在接下来的操作中帮助我们完成音名频率的输出
2.
其次是定时器的相关介绍,STM32F103C8T6有四个定时器,其中TIM1和TIM4为高级定时器,使用时需要TIM_CtrlPWMOutputs(TIM4,ENABLE); //MOE 主输出使能才能够正常使用。
STM32的定时器频率也可能不同,得看你的时钟是如何配置的,根据配置的不同,APB1总线上的时钟,如TIM2、TIM3、TIM4可能为36M,也可以为72M,对于不同的时钟频率,用同样的psc值配置会有不同的结果。

下面介绍音乐输出函数


```c
//蜂鸣器发出指定频率声音
void buzzerSound(unsigned short usFrep)
{
  GPIO_InitTypeDef GPIO_InitStr;
  unsigned long ulVal;
	if((usFrep<=1000000/65536UL)||(usFrep>20000))//1000000即100k,也就是上面所说
												//65535是计数器最大数值
	{
	  buzzerQuiet();//静音
	}
   else
	 {
		 ulVal=1000000/usFrep;
		 TIM4->ARR=ulVal;//设置自动重装载寄存器周期的值(音调)
		 TIM_SetCompare3(TIM4,ulVal/5);//设置比较值,调节占空比(音量)
		 TIM_Cmd(TIM4,ENABLE);//使能TIM4
	 }
}

```c
//蜂鸣器停止发声
void buzzerQuiet(void)
{
	TIM_Cmd(TIM4,DISABLE);//不使能TIM4
	GPIO_ResetBits(GPIOB,GPIO_Pin_8);//PB.8输出低
}


//演奏乐曲
void musicPlay(void)
{
	u8 i=0;
	while(1)
	{
	  if(MyScore[i].mTime==0)break;
		buzzerSound(MyScore[i].mName);
		delay_ms(MyScore[i].mTime);
		i++;
		buzzerQuiet();
		delay_ms(10);
	}

}

3 蜂鸣器参数

蜂鸣器
蜂鸣器的电压,阻值,还有尺寸都各不相同,在买蜂鸣器之前建议确定好尺寸再进行购买,一般情况下引脚间距是6-8mm,但是很有可能发生因为引脚间距不同而装不上的问题。比如有95.5这种小尺寸的,还有125.5的稍微大一些尺寸的,电压有3V,5V,12V的,电阻有16欧和42欧等等。一般来说,电阻越小,同等电压条件下的电流越大,其功率也就更高一些。

总结

其实有源蜂鸣器也能够发出音乐,不过因为有震荡源的干扰,音乐会有很多的杂音。
本文介绍了无源蜂鸣器和有源蜂鸣器,并介绍了相关乐理知识,以及如何将简谱改写为无源蜂鸣器结构体数组的相关知识。也进行了定时器时钟的arr和psc值的配置讲解。
如有错误,还请斧正。

  • 26
    点赞
  • 103
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 以下是基于STM32F103C8T6的5V有源蜂鸣器报警程序设计的代码示例: ```c #include "stm32f10x.h" void delay(unsigned int n) { while (n--) { for (int i = 0; i < 1000; i++); } } int main(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure); while (1) { GPIO_SetBits(GPIOB, GPIO_Pin_12); delay(1000); GPIO_ResetBits(GPIOB, GPIO_Pin_12); delay(1000); } } ``` 这个程序的作用是让蜂鸣器发出一段间隔为1秒的报警声音。在程序中,我们使用STM32F10x的GPIO库,通过设置GPIO的输出模式和电平来控制蜂鸣器的开关。同时,我们也使用了一个简单的延时函数来控制蜂鸣器的间隔时间。如果需要修改报警间隔,可以调整delay函数中的参数。 ### 回答2: 基于STM32F103C8T6的5V有源蜂鸣器报警程序设计如下: 首先,我们需要在STM32F103C8T6开发板上连接好5V有源蜂鸣器。将蜂鸣器的VCC引脚连接到STM32的5V电源引脚,GND引脚连接到STM32的GND引脚。另外,将蜂鸣器的信号引脚连接到STM32的一个GPIO引脚,例如PB0。 接下来,我们需要使用STM32的开发环境进行编程。首先,在程序中包含stm32f10x.h头文件。 然后,初始化相应的GPIO引脚。设置PB0引脚为输出模式,并将其输出低电平。 接下来,我们可以编写一个函数来控制蜂鸣器的报警。我们可以定义一个函数叫做Alarm(),用于开启或关闭蜂鸣器报警。 void Alarm(uint8_t enable) { if (enable) { GPIO_SetBits(GPIOB, GPIO_Pin_0); // 将PB0引脚输出高电平 } else { GPIO_ResetBits(GPIOB, GPIO_Pin_0); // 将PB0引脚输出低电平 } } 最后,在主函数中,我们可以通过调用Alarm()函数来触发蜂鸣器的报警。例如,我们可以让蜂鸣器在循环中每隔1秒钟开启和关闭一次: int main(void) { // 初始化GPIO的配置 GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure); while (1) { Alarm(1); // 开启蜂鸣器报警 delay(1000); // 延时1秒 Alarm(0); // 关闭蜂鸣器报警 delay(1000); // 延时1秒 } } 上述代码中,delay()函数可以根据具体的编程环境自行实现,用于延时一定的时间。以上就是基于STM32F103C8T6的5V有源蜂鸣器报警程序设计。 ### 回答3: 基于STM32F103C8T6的5V有源蜂鸣器报警程序设计可以通过以下步骤实现: 1. 了解STM32F103C8T6的引脚映射:根据资料,确定蜂鸣器连接的引脚,通常使用GPIO(通用输入/输出)引脚。 2. 配置GPIO引脚:使用STM32的开发环境,如Keil或CubeMX,选择正确的引脚作为输出,并进行初始化配置。 3. 编写控制代码:在主程序中,使用循环或条件语句,控制蜂鸣器的开启和关闭。可以将蜂鸣器的开关操作封装成函数,以便在需要的地方调用。 4. 设计报警模式:根据需求,设计合适的报警模式,如长鸣、短促间隔鸣叫、警报序列等。可以利用定时器中断或者延时函数来控制蜂鸣器的开启和关闭时间。 5. 测试和调试:将程序烧录到STM32F103C8T6开发板上,连接5V有源蜂鸣器,通过触发相关事件或条件,验证蜂鸣器是否按照预期工作。在调试过程中,可以使用串口或LED等辅助输出来观察程序执行状态。 需要注意的是,在设计报警程序时,应考虑系统资源利用和功耗控制等问题,确保蜂鸣器的报警功能稳定可靠。同时,根据需要可扩展其他功能,如报警触发条件、灵敏度调节等,实现更加多样化的应用场景。程序设计要确保稳定可靠,兼顾系统资源利用和性能优化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值