江科大51单片机笔记【10】蜂鸣器播放提示器音乐(下)

news2025/3/9 11:53:49

一、蜂鸣器播放提示器

这里我们要用Key,Delay,Nixie模块

并且把Nixie.c函数里的这两句注释,因为之前是动态显示,延时后马上清零,现在是静态显示,所以需要把他注释掉

//    Delay(1);
//    P0=0x00;

先验证一下模块是否正确代入

该现象为在数码管上显示按下按键的数字,复位就显示0

#include <REGX52.H>
#include " Delay.h"
#include " Key.h"
#include " Nixie.h"


unsigned char KeyNum;

void main()
{
	Nixie(1,0);
	while(1)
	{
 		KeyNum=Key();
		if(KeyNum)
		{
			Nixie(1,KeyNum);
		}
	
	}
}

上节课说过了,我们要让蜂鸣器的IO口翻转,翻转当然不止一次,要用到for循环,保证每翻转一次延时一段时间

之前我们写的Delay函数最短是延时1ms,但是标准的按键提示音一遍都是延时1000hz,也就是延时1us,所以我们在STC-ISP的软件延时计时器选择延时500us的函数

这样我们再把翻转的次数乘2,这样就是1000us了

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

//蜂鸣器端口
sbit Buzzer=P2^5;


/**
  * @brief  蜂鸣器私有延时函数,延时500us
  * @param  无
  
  * @retval 无

   */

void Buzzer_Delay500us()		//@11.0592MHz
{
	unsigned char i;
	i = 227;
	while (--i);
}


/**
  * @brief  蜂鸣器发声
  * @param  ms 发声的时长
  
  * @retval 无

   */

void Buzzer_Time(unsigned int ms)
{
	unsigned int i;
  for(i=0;i<ms*2;i++)
	{
		Buzzer=!Buzzer;
		Buzzer_Delay500us();	
	}

}

?/Buzzer.h

#ifndef __BUZZER_H__
#define __BUZZER_H__

void Buzzer_Time(unsigned int ms);

#endif

//main.c


#include <REGX52.H>
#include " Delay.h"
#include " Key.h"
#include " Nixie.h"
#include " Buzzer.h"

sbit Buzzer=P2^5;

unsigned char KeyNum;
unsigned int i;

void main()
{
	Nixie(1,0);
	while(1)
	{
 		KeyNum=Key();
		if(KeyNum)
		{
			Buzzer_Time(100);
			Delay(1);
		}
	
	}
}

这样第一个代码就完成啦

二、蜂鸣器音乐

这部分我们需要用到Delay和Timer0模块

1.取音符

我们定义一个数组导入上一节求出的重装载值,然后赋值给TL0和TH0,然后在中断函数里对Buzzer口取反,即每中断一次就取反一次

下面的代码是针对最低音的取值

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

sbit Buzzer=P2^5;

unsigned int FreqTable[]={
	63628,63731,63835,63928,64201,64103,64185,64260,64331,64400,64463,64528,
	64580,64633,64684,64732,64777,64820,64860,64898,64934,64968,65000,65030,
	65058,65085,65110,65134,65157,65178,65198,65217,65235,65252,65268,68283,
}

void main()
{
	Timer0Init();
	while(1)
	{
	}
}


void Timer0_Routine() interrupt 1
{
	TL0 = FreqTable[0]%256;  //设置定时初值
	TH0 = FreqTable[0]/256;  //设置定时初值
	Buzzer=!Buzzer;

}

2.根据时间取音符

我们先对照把音符对应的重装载值表做一个索引,即0-35

接着在看上图,把每个音符对应的重装载值的索引找出来,存在一个数组里,这样再定义一个变量,让它自加,这样我们就能选择到每个音符,也就是选择到重装载值,这样就能弹奏了

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

sbit Buzzer=P2^5;

unsigned int FreqTable[]={
	63628,63731,63835,63928,64201,64103,64185,64260,64331,64400,64463,64528,
	64580,64633,64684,64732,64777,64820,64860,64898,64934,64968,65000,65030,
	65058,65085,65110,65134,65157,65178,65198,65217,65235,65252,65268,68283,
};

unsigned char Music[]={12,12,19,19,21,21,19,17,17,16,16,14,14,12};

unsigned char FreqSelect,MusicSelect;


void main()
{
	Timer0Init();
	while(1)
	{
		FreqSelect=Music[MusicSelect];
		MusicSelect++;
		Delay(500);
	}
}


void Timer0_Routine() interrupt 1
{
	TL0 = FreqTable[FreqSelect]%256;  //设置定时初值
	TH0 = FreqTable[FreqSelect]/256;  //设置定时初值
	Buzzer=!Buzzer;

}

但实际操作起来发现旋律有了,但并不好听

这是因为我们相邻的音符直接只是加起来,而我们实际演奏时有手按下抬起的空余时间

我们就要模拟一个抬手的时间,我们在延时后面,关闭计时器,延时5ms,再打开,这样中间就有停顿了

然后我们发现有些音是两倍时间,我们又应该怎么控制每个音响的时间呢

因为分数不太好写代码,所以我们定义短的音符(如十六分音符)为基准,即1,长的音符即他的倍数,如四分音符就是4,全音符就是16

我们在前面定义音符索引的数组里,在每个索引后面再加上这个音符所对应的时长

我们前面定义4分音符是500ms,那么在这里16分音符就是125ms,为了方便修改,我们再定义一个变量存储这个500ms

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

sbit Buzzer=P2^5;

#define SPEED    500

unsigned int FreqTable[]={
	63628,63731,63835,63928,64201,64103,64185,64260,64331,64400,64463,64528,
	64580,64633,64684,64732,64777,64820,64860,64898,64934,64968,65000,65030,
	65058,65085,65110,65134,65157,65178,65198,65217,65235,65252,65268,68283,
};

unsigned char Music[]={
	12,4,
	12,4,
	19,4,
	19,4,
	21,4,
	21,4,
	19,4+4,
	17,4,
	17,4,
	16,4,
	16,4,
	14,4,
	14,4,
	12,4+4
};

unsigned char FreqSelect,MusicSelect;


void main()
{
	Timer0Init();
	while(1)
	{
		FreqSelect=Music[MusicSelect];
		MusicSelect++;
		Delay(SPEED/4*Music[MusicSelect]);
		MusicSelect++;
		TR0=0;
		Delay(5);
		TR0=1;
	}
}


void Timer0_Routine() interrupt 1
{
	TL0 = FreqTable[FreqSelect]%256;  //设置定时初值
	TH0 = FreqTable[FreqSelect]/256;  //设置定时初值
	Buzzer=!Buzzer;

}

到这里我们的小星星就可以演奏出来啦

3.进阶--天空之城

扩展:图中鼠标所指的0代表的是休止符,即代表在这期间是没有声音的

我们在之前的重装载值数组加上一个“0”代表休止符,放到第0位

对应的我们的音符索引表里的索引也就需要加1

然后我们的中断函数里加多一条判断,如果FreqTable[FreqSelect]=0,即重装载值不为0时才开始计时(演奏)

我们再加一个音乐终止符,即在最后让音乐停止,我们选择在音乐索引数组里,在最后加一个0xFF,然后在主函数的while里加上判断,只要当音乐索引不等于0xFF才进行操作

到这里就把我们的代码彻底完善了

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

sbit Buzzer=P2^5;

#define SPEED    500

unsigned int FreqTable[]={
	0,
	63628,63731,63835,63928,64201,64103,64185,64260,64331,64400,64463,64528,
	64580,64633,64684,64732,64777,64820,64860,64898,64934,64968,65000,65030,
	65058,65085,65110,65134,65157,65178,65198,65217,65235,65252,65268,68283,
};

unsigned char Music[]={
	13,4,
	13,4,
	20,4,
	20,4,
	22,4,
	22,4,
	20,4+4,
	18,4,
	18,4,
	17,4,
	17,4,
	15,4,
	15,4,
	13,4+4,
	
	0xFF
};

unsigned char FreqSelect,MusicSelect;


void main()
{
	Timer0Init();
	while(1)
	{
		if(Music[MusicSelect]!=0xFF)
		{
				FreqSelect=Music[MusicSelect];
				MusicSelect++;
				Delay(SPEED/4*Music[MusicSelect]);
				MusicSelect++;
				TR0=0;
				Delay(5);
				TR0=1;		
		}
		else
		{
			TR0=0;
			while(1);
		}
	}
}


void Timer0_Routine() interrupt 1
{
	if(FreqTable[FreqSelect])
	{
    	TL0 = FreqTable[FreqSelect]%256;  //设置定时初值
			TH0 = FreqTable[FreqSelect]/256;  //设置定时初值
			Buzzer=!Buzzer;
	}

}

下面我们开始写天空之城的代码

首先先把音符直接定义索引 ,因为天空之城比较多音符,我们要把36个音符都定义完

接下来对照着简谱,完善音乐数组,以音符+时长的方式写进去

写完之后我们运行一下,会发现报错数说内存过大。这是因为我们的单片机RAM:512字节,也就说我们的数组超过512个字节,那我们就要把他存到ROM:8K里,我们就需要再定义数组的时候

unsigned char code Music[],加一个关键字“code”就可以啦

但是这个数组就只变成了只读,里面的值就不能再更改

下面是完整代码

其实我们只需要通过修改音乐函数就可以演奏出歌曲啦

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

sbit Buzzer=P2^5;

#define SPEED    500


#define P    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



unsigned int  code FreqTable[]={
	0,
	63628,63731,63835,63928,64201,64103,64185,64260,64331,64400,64463,64528,
	64580,64633,64684,64732,64777,64820,64860,64898,64934,64968,65000,65030,
	65058,65085,65110,65134,65157,65178,65198,65217,65235,65252,65268,68283,
};


unsigned char code Music[]= {
	
  P,4,
  P,4,
	P,4,
	M6,2,
	M7,2,
	
	H1,4+2,
	M7,2,
	H1,4,
	H3,4,
	
	M7,4+4+4,
	M3,2,
	M3,2,
	
	M6,4+2,
	M5,2,
	M6,4,
	H1,4,
	
	M5,4+4+4,
	M3,4,
	
	M4,4+2,
	M3,2,
	M4,4,
	H1,4,
	
	M3,4+4,
	P,2,
	H1,2,
	H1,2,
	H1,2,
	
	M7,4+2,
	M4_,2,
	M4_,4,
	M7,4,
	
	M7,4+4,
	P,4,
	M6,2,
	M7,2,
	
	H1,2+4,
	M7,2,
	H1,4,
	H3,4,
	
	M7,4+4+4,
	M3,2,
	M3,2,
	
	M6,4+2,
	M5,2,
	M6,4,
	H1,4,
	
	M5,4+4+4,
	M2,2,
	M3,2,
	
	M4,4,
	H1,2,
	M7,4,
	H1,2+4,
	
	H2,2,
	H2,2,
	H3,2,
	H1,4+4,
	
	M5,4+4+4,
	M2,2,
	M3,2,
	
	M4,4,
	H1,2,
	H7,4,
	H1,2+4,
	
	H2,2,
	H2,2,
	H3,2,
	H1,4+4,
	
	H1,2,
	M7,4,
	M6,2,
	M6,2,
	M7,4,
	M5_,4,
	
	M6,4+4+4,
	H1,2,
	H2,2,
	
	H3,2+4,
	H2,2,
	H3,4,
	H5,4,
	
	H2,4+4+4,
	M5,2,
	M5,2,
	
	H1,2+4,
	M7,2,
	H1,4,
	H3,4,
	
	H3,4+4+4+4,
	
	M6,2,
	M7,2,
	H1,4,
	M7,4,
	H2,2,
	H2,2,
	
	H1,2+4,
	M5,2+4+4,
	
	H4,4,
	H3,4,
	H2,4,
	H1,4,
	
	H3,4+4+4,
	H3,4,
	
	H6,4+4,
	H5,4,
	H5,4,
	
	H3,2,
	H2,2,
	H1,4+4,
	P,2,
	H1,2,
	
	H2,4,
	H1,2,
	H2,2,
	H2,4,
	H5,4,
	
	H3,4+4+4,
	H3,4,
	
	H6,4+4,
	H5,4+4,
	
	H3,2,
	H2,2,
	H1,4+4,
	P,2,
	H1,2,
	
	H2,4,
	H1,2,
	H2,2+4,
	M7,4,
	
	M6,4+4+4,
	M6,2,
	M7,2,
 
	0xff,

};

unsigned char FreqSelect,MusicSelect;


void main()
{
	Timer0Init();
	while(1)
	{
		if(Music[MusicSelect]!=0xFF)
		{
				FreqSelect=Music[MusicSelect];
				MusicSelect++;
				Delay(SPEED/4*Music[MusicSelect]);
				MusicSelect++;
				TR0=0;
				Delay(5);
				TR0=1;		
		}
		else
		{
			TR0=0;
			while(1);
		}
	}
}


void Timer0_Routine() interrupt 1
{
	if(FreqTable[FreqSelect])
	{
    	TL0 = FreqTable[FreqSelect]%256;  //设置定时初值
			TH0 = FreqTable[FreqSelect]/256;  //设置定时初值
			Buzzer=!Buzzer;
	}

}

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

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

相关文章

Nginx1.19.2不适配OPENSSL3.0问题

Nginx 1.19.2 是较老的版本&#xff0c;而 Nginx 1.21 版本已经适配 OpenSSL 3.0&#xff0c;所以建议 升级 Nginx 到 1.25.0 或更高版本&#xff1a; wget http://nginx.org/download/nginx-1.25.0.tar.gz tar -xzf nginx-1.25.0.tar.gz cd nginx-1.25.0 ./configure --prefix…

蓝桥杯 Excel地址

Excel地址 题目描述 Excel 单元格的地址表示很有趣&#xff0c;它使用字母来表示列号。 比如&#xff0c; A 表示第 1 列&#xff0c; B 表示第 2 列&#xff0c; Z 表示第 26 列&#xff0c; AA 表示第 27 列&#xff0c; AB 表示第 28 列&#xff0c; BA 表示第 53 列&#x…

免费pdf格式转换工具

基本功能 - 支持单文件转换和批量转换两种模式 - 内置PDF文件预览功能 - 支持8种常见格式转换&#xff1a;Word、Excel、JPG/PNG图片、HTML、文本、PowerPoint和ePub 单文件转换功能 - 文件选择&#xff1a;支持浏览和选择单个PDF文件 - 输出位置&#xff1a;可自定义设置输出…

css错峰布局/瀑布流样式(类似于快手样式)

当样式一侧比较高的时候会自动换行&#xff0c;尽量保持高度大概一致&#xff0c; 例&#xff1a; 一侧元素为5&#xff0c;另一侧元素为6 当为5的一侧过于高的时候&#xff0c;可能会变为4/7分部dom节点 如果不需要这样的话删除样式 flex-flow:column wrap; 设置父级dom样…

Deepseek中的MoE架构的改造:动态可变参数激活的MoE混合专家架构(DVPA-MoE)的考虑

大家好,我是微学AI,今天给大家介绍一下动态可变参数激活MoE架构(Dynamic Variable Parameter-Activated MoE, DVPA-MoE)的架构与实际应用,本架构支持从7B到32B的等多档参数动态激活。该架构通过细粒度难度评估和分层专家路由,实现“小问题用小参数,大问题用大参数”的精…

docker-compose Install reranker(fastgpt支持) GPU模式

前言BGE-重新排名器 与 embedding 模型不同&#xff0c;reranker 或 cross-encoder 使用 question 和 document 作为输入&#xff0c;直接输出相似性而不是 embedding。 为了平衡准确性和时间成本&#xff0c;cross-encoder 被广泛用于对其他简单模型检索到的前 k 个文档进行重…

基于websocket的多用户网页五子棋 --- 测试报告

目录 功能测试自动化测试性能测试 功能测试 1.登录注册页面 2.游戏大厅页面 3.游戏房间页面 自动化测试 1.使用脑图编写web自动化测试用例 2.创建自动化项目&#xff0c;根据用例通过selenium来实现脚本 根据脑图进行测试用例的编写&#xff1a; 每个页面一个测试类&am…

在 macOS 上使用 CLion 进行 Google Test 单元测试

介绍 Google Test&#xff08;GTest&#xff09;是 Google 开源的 C 单元测试框架&#xff0c;它提供了简单易用的断言、测试夹具&#xff08;Fixtures&#xff09;和测试运行机制&#xff0c;使 C 开发者能够编写高效的单元测试。 本博客将介绍如何在 macOS 上使用 CLion 配…

深度解码!清华大学第六弹《AIGC发展研究3.0版》

在Grok3与GPT-4.5相继发布之际&#xff0c;《AIGC发展研究3.0版》的重磅报告——这份长达200页的行业圣经&#xff0c;不仅预测了2025年AI技术爆发点&#xff0c;更将「天人合一」的东方智慧融入AI伦理建构&#xff0c;堪称数字时代的《道德经》。 文档&#xff1a;清华大学第…

【论文笔记】Attentive Eraser

标题&#xff1a;Attentive Eraser: Unleashing Diffusion Model’s Object Removal Potential via Self-Attention Redirection Guidance Source&#xff1a;https://arxiv.org/pdf/2412.12974 收录&#xff1a;AAAI 25 作者单位&#xff1a;浙工商&#xff0c;字节&#…

97k倍区间

97k倍区间 ⭐️难度&#xff1a;中等 &#x1f31f;考点&#xff1a;暴力&#xff0c;2017省赛 &#x1f4d6; &#x1f4da; import java.util.Scanner;public class Main {static int N 100010;public static void main(String[] args) {Scanner sc new Scanner(System.…

cursor使用经验分享(java后端服务开发向)

前言 cursor是一款基于vscode&#xff0c;并集成AI能力的代码编辑器&#xff0c;其功能包括但不限于代码生成及补全、AI对话&#xff08;能够直接将代码环境作为上下文&#xff09;、即时应用建议等等&#xff0c;是一款面向未来的代码编辑器。 对于vscode&#xff0c;最先想…

SpringBoot3—场景整合:AOT

一、AOT与JIT AOT&#xff1a;Ahead-of-Time&#xff08;提前编译&#xff09;&#xff1a;程序执行前&#xff0c;全部被编译成机器码 JIT&#xff1a;Just in Time&#xff08;即时编译&#xff09;: 程序边编译&#xff0c;边运行&#xff1b; 编译&#xff1a;源代码&am…

蓝桥与力扣刷题(蓝桥 数字三角形)

题目&#xff1a; 上图给出了一个数字三角形。从三角形的顶部到底部有很多条不同的路径。对于每条路径&#xff0c;把路径上面的数加起来可以得到一个和&#xff0c;你的任务就是找到最大的和&#xff08;路径上的每一步只可沿左斜线向下或右斜线向下走&#xff09;。 输入描述…

游戏引擎学习第138天

仓库:https://gitee.com/mrxiao_com/2d_game_3 资产&#xff1a;game_hero_test_assets_003.zip 发布 我们的目标是展示游戏运行时的完整过程&#xff0c;从像素渲染到不使用GPU的方式&#xff0c;我们自己编写了渲染器并完成了所有的工作。今天我们开始了一些新的内容&#…

嵌入式学习L5D2-exec函数族和守护进程

exec函数族1 下面那个加了p环境变量就不用那个了。 输出的是系统 exec函数族2 后面不执行了 第二个参数瞎写也可以&#xff0c;但是要填 这里是说不想被替换&#xff0c;就在子进程里面执行这个。 守护进程概念 后台进程 守护进程是后台进程 一个fork了一个进程&#xff…

洛谷P1091

题目如下 思路 谢谢观看

行为模式---迭代器模式

概念 迭代器模式是设计模式的行为模式&#xff0c;它的主要设计思想是提供一个可以操作聚合对象&#xff08;容器或者复杂数据类型&#xff09;表示&#xff08;迭代器类&#xff09;。通过迭代器类去访问操作聚合对象可以隐藏内部表示&#xff0c;也可以使客户端可以统一处理…

【五.LangChain技术与应用】【29.LangChain Agent小案例1:智能代理的实战应用】

“为什么我的Agent总是处理不好实时数据?”“如何让AI自己调用API查股票?” 这些困扰开发者的问题,今天咱们用一个真实案例来彻底解决。不聊虚的,直接上手教你怎么用LangChain Agent造一个会自己查股价、算指标、生成报告的股票分析助手。全程高能,代码可直接复制粘贴到项…

windows部署spleeter 版本2.4.0:分离音频的人声和背景音乐

windows部署spleeter 版本2.4.0&#xff1a;分离音频的人声和背景音乐 一、Spleeter 是什么&#xff1f; Spleeter 是由法国音乐流媒体公司 Deezer 开发并开源的一款基于深度学习的音频分离工具。它能够将音乐中的不同音轨&#xff08;如人声、鼓、贝斯、钢琴等&#xff09;分…