【51单片机】10-蜂鸣器

news2024/12/23 23:56:24

1.蜂鸣器的原理

这里的“源”不是指电源。而是指震荡源。 也就是说,有源蜂鸣器内部带震荡源,所以只要一通电就会叫。 而无源内部不带震荡源,所以如果用直流信号无法令其鸣叫。必须用2K~5K的方波去驱动它。 有源蜂鸣器往往比无源的贵,就是因为里面含有震荡电路。

1.无源蜂鸣器原理

(1)早期的蜂鸣器都是无源的

(2)内部结构和材料

(3)发生原理:是利用了压电效应的原理

(4)控制信号,是高低电平相间

(5)电路图

(6)音调如何控制:音调受震动频率控制,就等于控制信号的频率。频率越高音调越高,听起来越刺耳。

(7)声音大小如何控制。由硬件决定的。无法写代码取控制声音大小。

2.有源蜂鸣器

1.无源蜂鸣器的缺陷

无源蜂鸣器只能在一定的范围内进行设置,太大或者太小都不行。【外部提供一个方波】

2.内置震荡电路后形成有源蜂鸣器

在内部添加震荡电路,不用使用外部的方波进行控制【但是内部的震荡电路是不可以改变的】

3.有源蜂鸣器也可以用频率信号驱动

可以直接加电也可以直接通过方波进行工作【有源蜂鸣器包含无源蜂鸣器】

4.三极管驱动

5.蜂鸣器发声音频率

2.让蜂鸣器响起来

1.接线确定

2.使用delay让蜂鸣器响起来

#include<reg51.h>


sbit BUZZER=P0^0;   //buzzer的驱动引脚



void delay(void){
	unsigned char i,j;
	for(i=0;i<100;i++){
		for(j=0;j<10;j++);
	}
	
}


void main(){
	
	while(1){
		BUZZER=1;//将引脚变外高电平
		delay();
		BUZZER=0;
		delay();
	}
}

 3.调节delay控制音乐的变化

控制delay中的i和j的大小,可以控制音调的振动频率,从而影响声音的尖锐。

如果i和j越大,则音调越小,声音越不尖锐

如果i和j越小,则音调越大,声音越尖锐

4.时钟周期的计算

3.用定时器控制蜂鸣器音调

1.上节驱动方法的问题

(1)不容易精确控制时间

(2)CPU控制蜂鸣器中不能做其他事

2.定时器控制蜂鸣器响

在一定时间后,通过定时器中的中断处理程序取减低蜂鸣器的电平,从控制蜂鸣器的响应

(1)10KHz=>1/10000s=>100us===>高电平+低电平的时间都为50us。所以要定的时间就是50us

(2)外部晶振12MHz+12T设置==>内部时钟频率1MHz===》1us===》TL0=255-50=205,TH0=255

3.注意点1:TL0和TH0的计算

我们在设置TH0和TL0应该将获得的时间取其补码

比如:我们是TL0=50; TH0=0;

        则我们应该写入“TL0=205;TH0=255;"

	TL0=205;
	TH0=0;
	TL0=205 % 256;//低位取余
	TH0=255/ 256;//高位取商

4.注意点2:定时器的计算

(65536-50000)/256

计数器是16位的,由高8位TH0和低8位TL0组成,可以存储2^16=65536个数,例如当设定计算值为65536-50000=15536时,也就是计数器从15536开始计时,到65536溢出,产生中断,对于晶振频率为12MHz的单片机来说,执行一个机器周期时长为1us,所以这里计时50000us,15536(D)转换为16进制是3CB0(H),此时TH0=3C,TL0=B0分别装入定时器即可,为了免除这些计算步骤,很多编程者采用"TH0=(65536-50000)/256;TL0=(65536-50000)%256",那么为什么要介入256呢?我们可以做一下运算,256(D)=0100(H),512(D)=0200(H),512(D)有两个256,所以高8位就是02,那么15536有多少个256?就是15536/256个,就相当于高8位有多少数值,商存入高8位,剩下的不足一个256,存入低8位,15536%256。

直接使用宏定义

【注意点】我们51是加法计数器,所以是65535-US

//宏定义一个时钟频率
#define XKHZ 10  //10*10的三次方Hz   要定多少Khz,就定义在这里
//宏定义us
//这里我们除以2,是因为想要分给TH0和TL0
//1000---》1000ms
#define US (1000/XKHZ)/2
sbit BUZZER=P0^0;   //buzzer的驱动引脚
//【注意点】因为51单片机是加法计数器,所以实际上我们要算的范围
//应该是65535-US,而不是0-US
#define N (65535 -US)

void delay(void){
	unsigned char i,j;
	for(i=0;i<100;i++){
		for(j=0;j<10;j++);
	}
	
}


void timer0_isr(void) interrupt 1 using 1{
	//这里再一次赋值,是因为我们想要他循环,所以我们要在他每一次进来的时候重新赋值
	
	TL0=N % 256;//低位取余
	TH0=N / 256;//高位取商
	BUZZER=!BUZZER;
	
}

4.蜂鸣器发出滴滴声音

通过count可以控制有声音和无声音的长短

#include<reg51.h>

/**

	用定时器控制蜂鸣器音调
*/
//宏定义一个时钟频率
#define XKHZ 10  //10*10的三次方Hz   要定多少Khz,就定义在这里
#define US (1000/XKHZ)/2
sbit BUZZER=P0^0;   //buzzer的驱动引脚
#define N (65535 -US)

//计数器
unsigned int count;

//判断此时是从”有声音“到"没声音”,还是从“没声音”到“有声音”
unsigned char flag=0;  //flag=0表示有声音,flag=1表示没有声音


void delay(void){
	unsigned char i,j;
	for(i=0;i<100;i++){
		for(j=0;j<10;j++);
	}
	
}


void timer0_isr(void) interrupt 1 using 1{
	//这里再一次赋值,是因为我们想要他循环,所以我们要在他每一次进来的时候重新赋值
	
	TL0=N % 256;//低位取余
	TH0=N / 256;//高位取商
	
	if(count--==0){
		//说明到了翻转的时候了
		//count=5000;//记得重新赋值,要不然就只能响一次
		if(flag==0){
			//说明之前处于有声音的,说明本次是从有声音到无声音的翻转
			flag=1;
            //此时“无声音”比“有声音”的时间还长了3倍
			count=600*3;
		}else{
			//说明之前没声音的,说明本次是没声音到有声音的翻转
			flag=0;
			//下面这句话,加上了,则表示在翻转的时候也会出现声音
			BUZZER=!BUZZER;
			count=600;
		}
	}else{//常规情况,也就是不翻转时候
		if(flag==0){
			BUZZER=!BUZZER; //4999次声音
		}else{
			//空的,因为不进行任何IO操作就是没有声音
		}
		
	}
	
	
}



void main(){
	
	//【第一步】初始化:我们使用的是定时器T0
	TMOD=0x01;  //T0使用16位bit定时器
	//********************************************************
	TL0=N % 256;//低位取余
	TH0=N / 256;//高位取商
	//********************************************************
	
	
	//打开计数器;TCON中的TR0【定时器T0的运行控制位】
	TR0=1;		//T0打开开始计数
	//T0的中断溢出位,表示允许中断
	ET0=1;		//T0中断允许
	EA=1;			//打开中断允许
	
	
	BUZZER=1;
	
	//设置响和不响的周期时间
	count=5000;     //5000*100us=500ms
	//初始化,有声音
	flag=0;
	
}

5.让蜂鸣器唱歌

1.为什么蜂鸣器可以唱歌

(1)发声音频可变---》延迟函数(delay)

(2)发声音长度可变---》定时器

unsigned char i;
for(i=0;i<200;i++){//控制声音响应时间长短
	Sound=~Sound;
	DelayXms(1);//控制声音的不同
}

for(i=0;i<50;i++){
	Sound=~Sound;
	DelayXms(2);
}

2.分析写好的唱歌程序

(1)复制代码过去

(2)修改控制蜂鸣器的IO引脚定义

3.”code“关键字的使用

因为我们加入的歌曲的编码是固定不变的,但是51单片机的内存有限制,所以我们只能把歌曲的编码放在常量区中,才使得其不会占据内存。则加上“code”关键字。

unsigned char code music_tab[] = 
{   
	0x18, 0x30, 0x1C , 0x10, //格式为: 频率常数, 节拍常数, 频率常数, 节拍常数,    
	0x20, 0x40, 0x1C , 0x10,   
	0x18, 0x10, 0x20 , 0x10,   
	0x1C, 0x10, 0x18 , 0x40,   
	0x1C, 0x20, 0x20 , 0x20,   
	0x1C, 0x20, 0x18 , 0x20,   
	0x20, 0x80, 0xFF , 0x20,   
	0x30, 0x1C, 0x10 , 0x18,   
	0x20, 0x15, 0x20 , 0x1C,
}

4.音节的构成

0x18, 0x30, 0x1C , 0x10,【2个一组】

(1)音调【振动频率决定】:0x18【奇数次】

(2)音长:0x30【偶数次】

注意点:

5.音频 VS 音调

(1)音节:定时器T0控制的是音乐的节拍(某一个音节持续时间)而不管音调(频率)

(2)音调【音频】:是直接使用delay做出来的,控制发出什么样子的声音


/************************************************************************  
[文件名]  C51音乐程序(八月桂花)  
[功能]    通过单片机演奏音乐  
  
/**********************************************************************/  

#include <REG51.H> 
//提供移位函数,可以省略
//#include <INTRINS.H>    
//本例采用89C52, 晶振为11.0592MHZ    
//关于如何编制音乐代码, 其实十分简单,各位可以看以下代码.    
//频率常数即音乐术语中的音调,而节拍常数即音乐术语中的多少拍;    
//所以拿出谱子, 试探编吧!    

sbit Beep =  P0^0 ; 			// 要根据实际的接线来修改
   
unsigned char n = 0;  //n为节拍常数变量    
unsigned char code music_tab[] = 
{   
	0x18, 0x30, 0x1C , 0x10, //格式为: 频率常数, 节拍常数, 频率常数, 节拍常数,    
	0x20, 0x40, 0x1C , 0x10,   
	0x18, 0x10, 0x20 , 0x10,   
	0x1C, 0x10, 0x18 , 0x40,   
	0x1C, 0x20, 0x20 , 0x20,   
	0x1C, 0x20, 0x18 , 0x20,   
	0x20, 0x80, 0xFF , 0x20,   
	0x30, 0x1C, 0x10 , 0x18,   
	0x20, 0x15, 0x20 , 0x1C,   
	0x20, 0x20, 0x20 , 0x26,   
	0x40, 0x20, 0x20 , 0x2B,   
	0x20, 0x26, 0x20 , 0x20,   
	0x20, 0x30, 0x80 , 0xFF,   
	0x20, 0x20, 0x1C , 0x10,   
	0x18, 0x10, 0x20 , 0x20,   
	0x26, 0x20, 0x2B , 0x20,   
	0x30, 0x20, 0x2B , 0x40,   
	0x20, 0x20, 0x1C , 0x10,   
	0x18, 0x10, 0x20 , 0x20,   
	0x26, 0x20, 0x2B , 0x20,   
	0x30, 0x20, 0x2B , 0x40,   
	0x20, 0x30, 0x1C , 0x10,   
	0x18, 0x20, 0x15 , 0x20,   
	0x1C, 0x20, 0x20 , 0x20,   
	0x26, 0x40, 0x20 , 0x20,   
	0x2B, 0x20, 0x26 , 0x20,   
	0x20, 0x20, 0x30 , 0x80,   
	0x20, 0x30, 0x1C , 0x10,   
	0x20, 0x10, 0x1C , 0x10,   
	0x20, 0x20, 0x26 , 0x20,   
	0x2B, 0x20, 0x30 , 0x20,   
	0x2B, 0x40, 0x20 , 0x15,   
	0x1F, 0x05, 0x20 , 0x10,   
	0x1C, 0x10, 0x20 , 0x20,   
	0x26, 0x20, 0x2B , 0x20,   
	0x30, 0x20, 0x2B , 0x40,   
	0x20, 0x30, 0x1C , 0x10,   
	0x18, 0x20, 0x15 , 0x20,   
	0x1C, 0x20, 0x20 , 0x20,   
	0x26, 0x40, 0x20 , 0x20,   
	0x2B, 0x20, 0x26 , 0x20,   
	0x20, 0x20, 0x30 , 0x30,   
	0x20, 0x30, 0x1C , 0x10,   
	0x18, 0x40, 0x1C , 0x20,   
	0x20, 0x20, 0x26 , 0x40,   
	0x13, 0x60, 0x18 , 0x20,   
	0x15, 0x40, 0x13 , 0x40,   
	0x18, 0x80, 0x00   
};   

 // T0定时控制的是音乐的节拍(某一个音节持续的时间)而不管音调(频率)
 // 音调是直接使用delay做出来的。
void int0()  interrupt 1   //采用中断0 控制节拍    
{  
		TH0 = 0xd8;   
   	TL0 = 0xef;   
   	n--;   
}   
   
void delay (unsigned char m)   //控制频率延时    
{   
		unsigned i = 3 * m;   
		while (--i);   
}   
   
void delayms(unsigned char a)  //豪秒延时子程序    
{   
  	while (--a);                  //采用while(--a) 不要采用while(a--); 各位可编译一下看看汇编结果就知道了!    
}   
   
void main()   
{ 
		unsigned char p, m;   // m为频率常数变量    
  	unsigned char i = 0;   

	//此处表示将:低4位留下了,高4位去除
  	TMOD &= 0x0f;   
  	TMOD |= 0x01;  		// timer0 工作在模式1,16位定时器下 
  	TH0 = 0xd8;
		TL0 = 0xef;   		// 这个TH和TL的值合起来定了1个10ms左右的一个时间
  	IE = 0x82;   
play:   
   	while (1)   
    {   
a: 		p = music_tab[i];   //表示指向第一个
			if (p == 0x00)   		// 一遍播放完了,延时1s后自动开始下一遍    
			{ 
				i=0;
				delayms(1000);  //表示播放完一次,延迟1s,接着下一次播放
				//如果想要播放一次,则下面goto注释
				goto play;  //跳转接着播放
			}     //如果碰到结束符,延时1秒,回到开始再来一遍    
     	else if (p == 0xff)  //0xff:休止符
			{ 
				i = i + 1;//跳过这一组数据
				delayms(100);
				TR0 = 0; //TR0:关闭定时器
				goto a;
			}  //若碰到休止符,延时100ms,继续取下一音符    
			else     //常规情况p==正常情况      
			{
				m = music_tab[i++];
				n = music_tab[i++];
			}  // m取频率常数【A,B,C】 和 n取节拍常数 【1/2,1/3,1/4】   
			
			//打开开定时器1   
				TR0 = 1;                               
				while (n != 0) //节拍不等于0
				{ 
					Beep = ~Beep;   //修改蜂鸣器电平 【~】与【!】一样
					delay(m);        	//等待节拍完成, 通过P1口输出音频(可多声道哦!)    
				}
					TR0 = 0;                         	//关定时器1    
			}   
}
  


6.切歌,暂停功能

单片机应用番外篇——蜂鸣器的应用之可实现切歌、暂停功能的简单音乐盒_哔哩哔哩_bilibili

/**

使用中断处理程序控制音乐的暂停和播放,切换


*/

//外部中断初始化

void EX_init(){
	
	IT0=1;//下降沿触发方式  INT0
	IT1=1;//下降沿触发方式    INT1
	EX0=1;  //外部中断0中断允许位
	EX1=1;   //打开中断开关
	EA=1;//打开总的中断开关
	
}

//暂停功能
void EX0_isr() interrupt 0
{
	DelayXms(10);//消除抖动
	if(INT0==0)//这里是INT1已经定义了,对应P3.2这个IO口,可以直接使用
	{
		//暂停通过定时计数器,则将计数器进行取反
		TR0=~TR0;
		TR1=~TR1;
	}
}

//切歌功能
void EX1_isr() interrupt 2
{
		DelayXms(10);//消除抖动
	if(INT1==0)//这里是INT1已经定义了,对应P3.2这个IO口,可以直接使用
	{
		state++;
		if(state==3){  //因为此时我们只有3曲歌
			state=0;
		}
	}
}

unsigned char state=0;//控制切歌

void main(){
	
	EX_init();
	InitialSound();
	while(1){
		switch(state){ //这个切歌,通过外部中断1,则进行转换
			case 0:
				Play(Music_Girl,0,2,345);break;
			case 1:
				Play(Music_haw,0,23,543);break;
			default:break;
		}
	}
	
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1050289.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

mysql双主互从

在主从配置的基础上&#xff0c;反向配置

zblog插件大全-zblog免费插件

在当今数字化时代&#xff0c;拥有一个精彩的博客或网站已经成为许多人追求的目标。通过博客&#xff0c;我们可以分享知识、表达观点&#xff0c;甚至创造收入。然而&#xff0c;维持一个充满新鲜内容的博客却不是一件容易的事情。 ZBlog自动采集插件 什么是ZBlog自动采集插件…

八、3d场景的区域光墙

在遇到区域展示的时候我们就能看到炫酷的区域选中效果&#xff0c;那么代码是怎么编辑的呢&#xff0c;今天咱们就好好说说&#xff0c;下面看实现效果。 思路&#xff1a; 首先&#xff0c;光墙肯定有多个&#xff0c;那么必须要创建一个新的js文件来作为他的原型对象。这个光…

IDEA的Maven换源

前言 IDEA是个好东西&#xff0c;但是使用maven项目时可能会让人很难受&#xff0c;要么是非常慢&#xff0c;要么直接下载不了。所以我们需要给IDEA自带maven换源&#xff0c;保证我们的下载速度。 具体操作 打开IDEA安装路径&#xff0c;然后打开下面的文件夹 plugins\m…

基于TOTP算法的Github两步验证2FA(双因子)机制Python3.10实现

从今年(2023)三月份开始&#xff0c;Github开始强制用户开启两步验证2FA(双因子)登录验证&#xff0c;毫无疑问&#xff0c;是出于安全层面的考虑&#xff0c;毕竟Github账号一旦被盗&#xff0c;所有代码仓库都会毁于一旦&#xff0c;关于双因子登录的必要性请参见&#xff1a…

利用Qt实现可视化科学计算器

&#x1f4de;个人信息 学号&#xff1a;102101433 姓名&#xff1a;林堂钦 &#x1f4a1; 作业基本信息 【课程】福州大学2021级软件工程Ahttps://bbs.csdn.net/forums/ssynkqtd-05作业要求链接https://bbs.csdn.net/topics/617294583作业目标 实现一个简易计算器&…

Ubuntu系统Linux内核安装和使用

安装&#xff1a; 检查树莓派Linux版本&#xff0c;我的是6.1 uname -r 内核下载链接&#xff1a; Raspberry Pi GitHub 找对应版本下载 导入之后&#xff0c;解压安装即可 unzip linux-rpi-6.1.y.zip 其他内容 treee 指令安装 sudo apt-get install tree 使用这…

一探Redis究竟:超火爆入门指南,你竟然还没看?

Redis入门教程目录&#xff1a;【Redis入门教程目录】 简介 Redis是由C语言编写的开源、基于内存、支持多种数据结构、高性能的Key-Value数据库。 特性 速度快 首先Redis是将数据储存在内存中的&#xff0c;通常情况下每秒读写次数达到千万级别。其次Redis使用ANSI C编写&…

No145.精选前端面试题,享受每天的挑战和学习

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6 🍨 阿珊和她的猫_CSDN个人主页 🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入…

LeetCode 377.组合总和IV 可解决一步爬m个台阶到n阶楼顶问题( 完全背包 + 排列数)

给你一个由 不同 整数组成的数组 nums &#xff0c;和一个目标整数 target 。请你从 nums 中找出并返回总和为 target 的元素组合的个数。 题目数据保证答案符合 32 位整数范围 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3], target 4 输出&#xff1a;7 解释&#x…

Spring源码篇(十)@Bean怎么注册一个bean

文章目录 前言配置类里的Bean解析sourceClass是什么解析Bean方法添加Bean注解的方法信息注册总结Bean注册的过程注意点 前言 配置类的解析之前有聊过&#xff0c;这篇也会涉及到一部分&#xff0c;因为Bean本身也是配置类里的一个东西&#xff0c;本篇会着重解析Bean注册bean的…

区块链(6):p2p去中心化介绍

1 互联网中中心化的服务和去中心化服务的概念介绍 目前的互联网公司大都是中心化的 区块链网络大多是去中心化的 去中心化 2 p2p的简单介绍 java 网络编程:socket编程,netty编程,websoket简单介绍 2.1 节点是如何提供服务的(web编程实现)

qml保姆级教程一:布局组件

&#x1f482; 个人主页:pp不会算法v &#x1f91f; 版权: 本文由【pp不会算法v】原创、在CSDN首发、需要转载请联系博主 &#x1f4ac; 如果文章对你有帮助、欢迎关注、点赞、收藏(一键三连)和订阅专栏哦 QML系列教程 QML教程一&#xff1a;布局组件 文章目录 锚布局anchors属…

前端开发 vs. 后端开发:编程之路的选择

文章目录 前端开发&#xff1a;用户界面的创造者1. HTML/CSS/JavaScript&#xff1a;2. 用户体验设计&#xff1a;3. 响应式设计&#xff1a;4. 前端框架&#xff1a; 后端开发&#xff1a;数据和逻辑的构建者1. 服务器端编程&#xff1a;2. 数据库&#xff1a;3. 安全性&#…

CISSP学习笔记:人员安全和风险管理概念

第二章 人员安全和风险管理概念 2.1 促进人员安全策略 职责分离: 把关键的、重要的和敏感工作任务分配给若干不同的管理员或高级执行者&#xff0c;防止共谋工作职责:最小特权原则岗位轮换:提供知识冗余&#xff0c;减少伪造、数据更改、偷窃、阴谋破坏和信息滥用的风险&…

CSS详细基础(四)显示模式

本帖开始介绍CSS中更复杂的内容 目录 一.显示模式 1.行内元素 2.块级元素 3.行内块元素 二.背景样式 一.显示模式 顾名思义&#xff0c;在CSS中&#xff0c;元素主要有3种显示模式&#xff1a;行内元素、块级元素、行内块元素~ 所谓块级元素&#xff0c;指的是该元素在…

Springboot+vue的企业人事管理系统(有报告),Javaee项目,springboot vue前后端分离项目。

演示视频: Springbootvue的企业人事管理系统&#xff08;有报告&#xff09;&#xff0c;Javaee项目&#xff0c;springboot vue前后端分离项目。 项目介绍&#xff1a; 本文设计了一个基于Springbootvue的前后端分离的企业人事管理系统&#xff0c;采用M&#xff08;model&am…

【kylin】【ubuntu】搭建本地源

文章目录 一、制作一个本地源仓库制作ubuntu本地仓库制作kylin本地源 二、制作内网源服务器ubuntu系统kylin系统 三、使用内网源ubuntukylin 一、制作一个本地源仓库 制作ubuntu本地仓库 首先需要构建一个本地仓库&#xff0c;用来存放软件包 mkdir -p /path/to/localname/pac…

基于微信小程序的手机在线商城小程序设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言系统主要功能&#xff1a;具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计…

摩根大通限制英国客户购买加密货币,市场掀起涟漪!

摩根大通旗下英国数字银行部门宣布&#xff0c;从下个月开始&#xff0c;将禁止客户进行加密货币交易。这一决定归因于人们越来越担心与加密货币相关的诈骗和欺诈行为的增加。 正如该银行周二表示的那样&#xff0c;从10月16日起&#xff0c;该银行的客户将不再可以选择通过借…