51单片机学习笔记-9蜂鸣器

这篇具有很好参考价值的文章主要介绍了51单片机学习笔记-9蜂鸣器。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

9 蜂鸣器

[toc]

注:笔记主要参考B站江科大自化协教学视频“51单片机入门教程-2020版 程序全程纯手打 从零开始入门”。
注:工程及代码文件放在了本人的Github仓库。


9.1 蜂鸣器简介

蜂鸣器 是一种将电信号转换为声音信号的器件,常用来产生设备的按键音、报警音等提示信号。蜂鸣器按驱动方式可分为有源蜂鸣器和无源蜂鸣器:

有源蜂鸣器:内部自带振荡源,将正负极接上直流电压即可持续发声,频率固定。
无源蜂鸣器:内部不带振荡源,需要控制器提供振荡脉冲(一般是PWM波)才可发声,调整提供振荡脉冲的频率,可发出不同频率的声音。
注:两者外观几乎没有差别,购买的时候要注意。
再注:单片机上自带的蜂鸣器为无源蜂鸣器

单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-1 蜂鸣器实物图

由于单片机引脚的驱动能力弱,所以一般不能直接用于驱动信号,而是作为开关信号来控制驱动能力更强的信号,从而驱动元器件正常工作。驱动蜂鸣器的方式主要有两种:三极管驱动(图8-2)、集成电路驱动(图8-3)。单片机采用集成电路芯片 达林顿晶体管阵列 ULN2003D 驱动蜂鸣器。

单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-2 蜂鸣器-三极管驱动方式
单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-3 蜂鸣器-集成电路驱动方式

达林顿管又称复合管。他将两个三极管串联,以组成一只等效的新的三极管。这只等效三极管的放大倍数是原二者之积,因此它的特点是 放大倍数非常高达林顿管的作用一般是在高灵敏的放大电路中放大非常微小的信号,如大功率开关电路。在电子学电路设计中,达林顿接法常用于功率放大器和稳压电源中。达林顿晶体管相当于非门,但是输出的“1”是高阻态,没有驱动能力;输出的“0”则有驱动能力。

单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-4 ULN2003逻辑框图

下面来分析乐理。无源蜂鸣器只有在由通路变成断路时才会发出声音,反之由断路变成通路并不发出声音。根据电路原理图可知,只要控制P2_5下降沿的频率就可以控制蜂鸣器发出不同的声音,而调节占空比则可以控制蜂鸣器的音量。下图8-5中,按照一组音调依次分成4组:大字组、小字组、小字1组、小字2组。组与组之间相差“8度”;相邻的两个键之间(包括白键与黑键)相差半音; 数字简谱左上标“#”表示升高半音,“b”表示降低半音,升降号在本小节有效。 下图8-6给出了“小字组、小字1组、小字2组”音符与频率的对应关系。也就是说,每一个音符都对应了一个特定的频率。
而为了按照正常的节奏演奏出整首音乐,还需要调整每个音符所占用的时间。下图8-7给出了不同的音符的时值比,可以看出在音符的右侧和下面加横线可以实现2的幂次的调整,而在右侧加点则表示乘以1.5倍。 一般设置四分音符的持续时间为500ms。
于是根据下面两张图,整体思路如下:钢琴五线谱→数字简谱→蜂鸣器频率。 通过不断改变定时器的复位值,进而控制蜂鸣器的频率变化。

单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-5 钢琴谱与数字简谱的转换关系
单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-6 数字简谱与蜂鸣器频率对应关系
单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-7 不同的音符的时值比

9.2 实验:蜂鸣器播放提示音

需求:对于四个独立按键开关,每当按键按下,蜂鸣器给出提示音,并且在数码管上显示按下的按键值。

本实验使用软件延时,完成蜂鸣器的频率调整和蜂鸣持续时间。代码如下:

- main.c

#include <REGX52.H>
#include "PushKey.h"
#include "NixieTube.h"
#include "Buzzer.h"

void main(){
  unsigned char key_num = 0;
  NixieTube(8,key_num); // 数码管显示初始化
  while(1){
    key_num = PushKey();
    if(key_num){
      NixieTube(8,key_num);
      Buzzer_Time(500);
    }
  }
}

- Buzzer.h

#ifndef __BUZZER_H__
#define __BUZZER_H__

void Buzzer_Time(unsigned int ms);//1000Hz,持续ms个毫秒

#endif

- Buzzer.c

#include <REGX52.H>

// 重新定义端口名
sbit Buzzer_con = P2^5;

/**
  * @brief :蜂鸣器专用延时函数,@11.0592MHz晶振,延时500us
  * @param :需要延时500us的个数
  * @retval :无
 */
void Buzzer_Delay100us(unsigned int cycles){
	unsigned char i;
  do{
	i = 43;
	while (--i);
  }while(--cycles);
}


/**
  * @brief :蜂鸣器以默认频率1000Hz,持续响指定时间。
  * @param :cycles持续时间(响cycles个1ms)。
  * @retval :无
 */
void Buzzer_Time(unsigned int ms){
  unsigned int i;
  for(i=0;i<ms;i++){
    Buzzer_con = !Buzzer_con;//每500us翻转一次
    Buzzer_Delay100us(5);//控制蜂鸣器频率为1000Hz
  }
}

- PushKey.h

#ifndef __PUSHKEY_H__
#define __PUSHKEY_H__

// 延时cycles ms,晶振@11.0592MHz
void PushKey_Delay(unsigned char cycles){
  unsigned char i, j;
  do{
    i = 2;
    j = 199;
    do{
      while (--j);
    }while (--i);
  }while(--cycles);
}

/**
  * @brief :检测按下了哪个按键开关
  * @param :无
  * @retval :输出按键开关编号1~4,不按返回0,松开触发
 */
unsigned char PushKey(){
  unsigned char key = 0;
  if(!P3_1)     {PushKey_Delay(10);while(!P3_1);PushKey_Delay(10);key=1;}
  else if(!P3_0){PushKey_Delay(10);while(!P3_0);PushKey_Delay(10);key=2;}
  else if(!P3_2){PushKey_Delay(10);while(!P3_2);PushKey_Delay(10);key=3;}
  else if(!P3_3){PushKey_Delay(10);while(!P3_3);PushKey_Delay(10);key=4;}
  return key;
}

#endif

- NixieTube.h

#ifndef __NIXIETUBE_H__
#define __NIXIETUBE_H__

void NixieTube(unsigned char led, unsigned char num){
  // 给出数字0~9的定义(符合数组的索引)
  unsigned char number[10] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
  // 给出选择的LED1~LED8的定义(实际上只用P2的2/3/4引脚)
  unsigned char sel_led[8] = {0x00,0x04,0x08,0x0c,0x10,0x14,0x18,0x1c};
  P0 = 0x00; // 数码管不显示,消影
  P2 = sel_led[led-1]; // 选择数码管:LED1
  P0 = number[num];  // 数码管显示7
}

#endif

9.3 实验:蜂鸣器播放音乐

需求:使用蜂鸣器播放小星星、天空之城。

单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-8 《小星星》曲谱
单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-9 《天空之城》曲谱

由于单片机性能有限,实现播放音乐的功能之后几乎就没什么资源了,所以就不进行模块化了。为了便捷的控制蜂鸣器频率,使用定时器T0,通过对其溢出复位值得调整,以控制蜂鸣器周期。下面是代码的调用逻辑:

单片机蜂鸣器,# 51单片机-江科大,51单片机,学习,单片机
图9-10 “蜂鸣器播放音乐”的代码调用关系

代码展示:

- main.c

#include <REGX52.H>
#include "Timer0.h"

// 下面的乐谱可以替换
/***************************************************/
// 《小星星》乐谱定义
#define  music_len 42//指定乐谱长度
unsigned char code music_freq_sel[2][music_len] = {
{12,12,19,19, 21,21,19,    17,17,16,16, 14,14,12,    19,19,17,17, 16,16,14,   
 19,19,17,17, 16,16,14,    12,12,19,19, 21,21,19,    17,17,16,16, 14,14,12   },
{4,4,4,4, 4,4,8, 4,4,4,4, 4,4,8, 4,4,4,4, 4,4,8,
 4,4,4,4, 4,4,8, 4,4,4,4, 4,4,8, 4,4,4,4, 4,4,8}};
/***************************************************/

// 通用延时函数,1ms
void Delay1ms(unsigned int cycles){
  unsigned char i, j;
  do{
    i = 2;
    j = 199;
    do{
      while (--j);
    }while (--i);
  }while(--cycles);
}

// 重新定义端口名
sbit Buzzer_con = P2^5;

// 定义四分音符时长(500ms)
#define SPEED 500

//不同频率所对应的定时器重装载值
unsigned int freq_reload[37] =
{0,63628,63731,63835,63928,64021,64103,64185,64260,64331,64400,64463,64524,
   64580,64633,64684,64732,64777,64820,64860,64898,64934,64968,65000,65030,
   65058,65085,65110,65134,65157,65178,65198,65217,65235,65252,65268,65283};
//定义所需要选择的频率索引
unsigned char freq_sel = 0;
//定义音乐的索引变换,时长变换(时长以1/16音符为基准)
#define N0  0
#define L1  1
#define L1_ 2
#define L2  3
#define L2_ 4
#define L3  5
#define L4  6
#define L4_ 7
#define L5  8
#define L5_ 9
#define L6  10
#define L6_ 11
#define L7  12
#define M1  13
#define M1_ 14
#define M2  15
#define M2_ 16
#define M3  17
#define M4  18
#define M4_ 19
#define M5  20
#define M5_ 21
#define M6  22
#define M6_ 23
#define M7  24
#define H1  25
#define H1_ 26
#define H2  27
#define H2_ 28
#define H3  29
#define H4  30
#define H4_ 31
#define H5  32
#define H5_ 33
#define H6  34
#define H6_ 35
#define H7  36

void main(){
  unsigned int i;
  //定时器T0初始化
  Timer0_Init();
  freq_sel = 0;
  while(1){
    for(i=0;i<music_len;i++){
      freq_sel = music_freq_sel[0][i];
      Delay1ms(SPEED/4*music_freq_sel[1][i]);
      TR0 = 0;
      Delay1ms(5);
      TR0 = 1;
    }
    TR0 = 0;
    Delay1ms(5000); //暂停一段时间
    TR0 = 1;
  }
}

// 定义定时器T0中断后要执行的动作
void Timer0_Routine() interrupt 1{
  //当频率不是“0”,才进行相应的振荡
  if(freq_reload[freq_sel]){
    TH0 = freq_reload[freq_sel]/256;
    TL0 = freq_reload[freq_sel]%256;
    Buzzer_con = !Buzzer_con;
  }  
}

- Timer0.h

#ifndef __TIMER0_H__
#define __TIMER0_H__

#include <REGX52.H>
/**
  * @brief :对定时器0进行初始化,初始化完成后定时器0即可正常工作。
  * 注:对11.0592MHz进行12分频(脉冲周期1.0850694us)。
  * 注:配置过程中,由于掉电复位后中断都默认不开启,所以只需配置定时器0
  *     相关的寄存器即可,不要定义其他中断的寄存器,以保证程序的复用性。
 */
void Timer0_Init(){
  // 配置定时器T0的相关寄存器
  TMOD&=0xf0; TMOD|=0x01; // 选择T0的GATE=0/允许计数/模式1
  // 上面这个方法目的是不干扰高四位,对低四位先清零再加值。
  TF0 = 0; TR0 = 1; // 溢出标志位清空,运行控制位置1
  TH0 = 0xfc; TL0 = 0x66; // 离溢出近似1ms
  // 注:上面这个初值只在第一次溢出生效,后面都是从0开始计数。
  // 配置中断寄存器
  EA = 1; ET0 = 1; // 不屏蔽所有中断,允许T0溢出中断
  PT0 = 0; // T0优先级保持默认,不写这句话也可以
}

/*中断函数模板
// 定义定时器T0中断后要执行的动作
void Timer0_Routine() interrupt 1{
  static unsigned int count_T0; //中断次数
  count_T0++; //更新中断次数
  TH0 = 0xfc; TL0 = 0x66; // 恢复溢出周期,近似1ms
  if(count_T0>500){
    count_T0 = 0;
    
  }  
}
*/
#endif

编程感想:文章来源地址https://www.toymoban.com/news/detail-736915.html

  1. 关于位运算。在FPGA中经常使用位运算提升运算效率,即使位宽不同也可以很方便的指定位的索引;但是在单片机中,运算中通常不能指定位的索引,所以在进行不同位宽的位运算时,会出现一些奇怪的bug。即,在单片机中,除了一些特殊的场合(如寄存器赋值),应避免使用复杂的位运算。
  2. 愚蠢的错误:音乐播放完之后,不会继续播放,而是持续的嘈杂。原因:定义音乐的数组长度太长了,导致最后的音是空的。即数组的溢出错误,代码飘到哪里就不知道了。
  3. 下面是其他音乐的一些乐谱:
/***************************************************/
// 《天空之城》曲谱
#define  music_len 121//指定乐谱长度
unsigned char code music_freq_sel[2][music_len] = {
{ N0,N0,N0,M6,M7,  H1,M7,H1,H3,  M7,M3,M3,
  M6,M5,M6,H1,  M5,M3,  M4,M3,M4,H1,
  M3,N0,H1,H1,H1, M7,M4_,M4_,M7,  M7,N0,M6,M7,
  H1,M7,H1,H3,  M7,M3,M3,  M6,M5,M6,H1,
  M5,M2,M3,  M4,H1,M7,H1,  H2,H2,H3,H1,
  H1,M7,M6,M6,M7,M5_,  M6,H1,H2,  H3,H2,H3,H5,
  H2,M5,M5,  H1,M7,H1,H3,  H3,
  M6,M7,H1,M7,H2,H2,  H1,M5,  H4,H3,H2,H1,
  H3,H3,  H6,H5,H5,  H3,H2,H1,N0,H1,
  H2,H1,H2,H2,H5,  H3,H3,  H6,H5,
  H3,H2,H1,N0,H1,  H2,H1,H2,M7,  M6,M6,M7},
{ 4,4,4,2,2,  6,2,4,4,  12,2,2,
  6,2,4,4,  12,4,  6,2,4,4,
  8,2,2,2,2,  6,2,4,4,  8,4,2,2,
  6,2,4,4,  12,2,2,  6,2,4,4,
  12,2,2,  4,2,4,4, 2,2,2,8,
  2,2,2,2,4,4,  12,2,2,  6,2,4,4,
  12,2,2,  6,2,4,4,  16,
  2,2,4,4,2,2,  6,10,  4,4,4,4,
  12,4,  8,4,4,  2,2,8,2,2,
  4,2,2,4,4,  12,4,  8,8,
  2,2,8,2,2,  4,2,6,4,  12,2,2}
};
/***************************************************/

到了这里,关于51单片机学习笔记-9蜂鸣器的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【【51单片机的蜂鸣器发声】】

    我们在设置板子的时候要翻转 翻转IO口是因为使用了步进电机的多余的芯片管脚 我们翻转之后还想让它保持一段时间 就是先翻转500次 Delay(1)之后 就是最小的单位每隔1ms 周期就是2ms 频率就是500HZ

    2024年02月16日
    浏览(51)
  • 【【51单片机蜂鸣器实现起风了】】

    前面的有两个非常关键的点希望兄弟们明白 我一开始也失算了 这里兄弟们注意务必改成int 不然会超出 就会出现播放一半从头开始的情况 下面是两份起风了代码直接附上main.c 因为另外的其实和我之前说的模板都一样复制粘贴就行 为什么是2份因为什么ABCD大调我不懂啊 我看乐

    2024年02月16日
    浏览(94)
  • 【【51单片机的蜂鸣器实现小星星】】

    我们在设置板子的时候要翻转 翻转IO口是因为使用了步进电机的多余的芯片管脚 我们翻转之后还想让它保持一段时间 就是先翻转500次 Delay(1)之后 就是最小的单位每隔1ms 周期就是2ms 频率就是500HZ 蜂鸣器播放音乐 我们首先在前面到现在必须明白一个道理 就是我们的蜂鸣器发

    2024年02月16日
    浏览(32)
  • stm32单片机/51单片机蜂鸣器不响(proteus模拟)

    1提高蜂鸣器电阻,这样根据分压原理,可以提升蜂鸣器2段电压 2更改蜂鸣器的工作电压为更小的值,这个可以通过在proteus内设置探针查看电压值,更改为其值就可以了 我觉得如果不是仿真的话,蜂鸣器额定电压是不能改的,如果电压不够的话好像就只能够选用合适的三极管

    2024年02月11日
    浏览(29)
  • MCU-51:单片机蜂鸣器播放音乐和提示音

    蜂鸣器是一种将电信号转换为声音信号的器件,常用来产生设备的按键音、报警音等提示信号 蜂鸣器按驱动方式可分为有源蜂鸣器和无源蜂鸣器 有源蜂鸣器:内部自带振荡源,将正负极接上直流电压即可持续发声,频率固定 无源蜂鸣器:内部不带振荡源,需要控制器提供振

    2024年02月02日
    浏览(34)
  • 学习笔记|认识蜂鸣器|控制原理|电磁炉LED实战|逻辑运算|STC32G单片机视频开发教程(冲哥)|第八集(上):蜂鸣器应用

    反面包上黑胶的是有源蜂鸣器。 在背面裸露的一块线路板的是无源蜂鸣器。 YX55675-无源蜂鸣器模块的资料 提取码:nl73 1、有源蜂鸣器内部带震荡源,所以只要一通电就会叫(一边高电平,一边低电平),而无源内部不带震荡源,所以如果用直流信号无法令其鸣叫(必须不断给高

    2024年02月11日
    浏览(36)
  • 33、基于51单片机老人防跌倒蜂鸣器报警系统加速度检测

    背景技术 老年人出门由于身体不灵活、视力较差,容易发生跌倒,现用的老年人跌倒报警装置是通过无线对讲系统研发的,它外观精美,自动化程度高,有很强的专业性,但是,设计者忽略了一个问题,因为大多数老年人使用现代化产品不熟练,随着视力、操作能力的下降,

    2023年04月08日
    浏览(70)
  • 【51单片机】:闪灯、流水灯(从左往右)、流水灯(从右往左)、蜂鸣器。

            言归正传,一切都回归本源,从51再来一次,首先就是最简单的点灯大师。         这么些年从51 到  32  再到  arduino  再到  ESP  等等,都是从点灯大师开始的。 于是介绍如下: 闪灯 流水灯(从左往右) 流水灯(从右往左) 蜂鸣器         LED灯的原理:LED是发

    2024年02月11日
    浏览(37)
  • 单片机学习-蜂鸣器如何发出声音

    BEEP =! BEEP    为何使用 非运算,这里面 P2 的管脚默认 有上拉电阻,P2^5 输出高电平 每次循环 非运算一下 为低电平 ,产生那种评率 0 1  0 1  0 1 循环下去 

    2024年02月11日
    浏览(29)
  • 单片机学习-蜂鸣器电子元件

    蜂鸣器是有什么作用的? 蜂鸣器 是 一种 一体化结构 的电子训响器,可以发出声音的电子元器件 蜂鸣器分类? ①压电式蜂鸣器(图左) 称: 无源蜂鸣器 ②电磁式蜂鸣器(图右) 称: 有源蜂鸣器  组成: ①压电式蜂鸣器主要由多谐振荡器、压电蜂鸣片、阻抗匹配器及共鸣

    2024年02月11日
    浏览(29)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包