[Arduino Uno 实验]使用Arduino Uno开发板与无源蜂鸣器播放乐曲

2024-01-15 03:08

本文主要是介绍[Arduino Uno 实验]使用Arduino Uno开发板与无源蜂鸣器播放乐曲,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 前言
  • 一、无源蜂鸣器电路
  • 二、MIDI文件转换
  • 三、实例分析


前言

本实验结合python程序转换,利用Arduino开发板配合无源蜂鸣器电路播放任意MIDI乐曲。

MIDI:MIDI是编曲界最广泛的音乐标准格式,可称为“计算机能理解的乐谱”。它用音符的数字控制信号来记录音乐。一首完整的MIDI音乐只有几十KB大,而能包含数十条音乐轨道。

无源蜂鸣器:无源蜂鸣器利用电磁感应现象,为音圈接入交变电流后形成的电磁铁与永磁铁相吸或相斥而推动振膜发声,接入直流电只能持续推动振膜而无法产生声音,只能在接通或断开时产生声音。


一、无源蜂鸣器电路

无源蜂鸣器电路示意图

如图所示,连接好电路,并将8号端点连接Arduino Uno开发板上的8号数字I/O端口。

我们知道,Arduino Uno开发板的数字I/O端口通过PWM1的方式进行模拟信号输出,无源蜂鸣器在外加直流电压的前提下接受该模拟信号即可发出相应频率的声音。

二、MIDI文件转换

 对于任意MIDI音频文件,我们可以提取其各个轨道的信息,经过适当的转换后得到各个音的频率与发音时间,再结合Arduino开发板的一些功能,就可以利用蜂鸣器播放相应的音乐了。

 MIDI文件转换程序将MIDI文件转换为cpp格式,将转换后的cpp代码粘贴到ArduinoUno IDE并上传到开发板即可运行。转换程序支持设定播放乐曲的目标速率(bpm)。

转换程序的代码如下(有缺省):

import mido
import time
from alive_progress import alive_barmid = mido.MidiFile()  # midi文件
file_name = ""  # 文件名
bpm = 0  # 速率
track_count=0#轨道数
note_count=0#音符数note_list = []  # 频率表
dura_list = []  # 延时表#88键频率对照表
fre_list = []#此处省略具体内容def tick2time(bpm, tick):'''根据bpm计算音符持续时间(ms)'''t = 60*1000/bpm  # 拍时return round(tick/mid.ticks_per_beat*t)def create_cpp():'''由midi文件生成对应cpp文件'''global track_count,note_count,note_list, dura_list  with alive_bar(len(mid.tracks)) as bar:for track in mid.tracks:bar()for i, msg in enumerate(track):if msg.type == "note_on":if i == len(track)-1 or track[i+1].type == "note_off":if msg.time != 0:note_list.append(0)dura_list.append(tick2time(bpm, msg.time))note_list.append(fre_list[msg.note-21])elif msg.type == "note_off":if i == len(track)-1 or track[i+1].type == "note_on":dura_list.append(tick2time(bpm, msg.time))with open(file_name.split('.')[0]+f'--track_{track_count}'+'.cpp', 'w') as f:f.write(f'\nint melody[{max(len(dura_list),len(note_list))+1}]='+str(note_list).replace('[', '{').replace(']', '}')+';')f.write(f'\nint noteDurations[{max(len(dura_list),len(note_list))+1}]='+str(dura_list).replace('[', '{').replace(']', '}')+';')f.write('\nvoid setup() {')f.write('\n}')f.write('''\nvoid loop(){ for(int thisNote = 0; thisNote < sizeof(melody)/4; thisNote ++){tone(8,melody[thisNote],noteDurations[thisNote]);delay(noteDurations[thisNote]);noTone(8);}}''')track_count+=1note_count+=len(note_list)note_list = []  dura_list = []  if __name__ == "__main__":print("========================================================")print("Program started.\n")file_name = input('please input the file name (same path):')bpm = int(input('please input the playing speed (bpm):'))print('\n\nProcessing...\n')mid = mido.MidiFile(file_name)  # 打开midi文件create_cpp()print(f'\n\nCompleted.  \n{track_count} tracks found. \n{note_count} notes created in total.')print("\n========================================================")input("\nPress any key to exit...")

三、实例分析

 以一首Astronomia为例,将文件拷贝到转换程序同一目录下,运行转换程序,输入“Astronomia.mid”与“80”(bpm),等待程序运行,得到结果。

转换结果如下(Astronomia,共266个音符):


int melody[267]={0, 523, 466, 0, 440, 349, 392, 0, 392, 0, 587, 0, 523, 0, 466, 0, 440, 0, 440, 0, 523, 0, 466, 0, 440, 392, 0, 392, 0, 932, 880, 932, 880, 932, 392, 0, 392, 932, 880, 932, 880, 932, 392, 0, 392, 587, 523, 0, 466, 0, 440, 0, 440, 440, 0, 523, 0, 466, 0, 440, 0, 392, 0, 392, 0, 932, 880, 932, 0, 880, 932, 392, 0, 392, 932, 0, 880, 932, 0, 880, 932, 392, 0, 392, 587, 0, 523, 0, 466, 0, 440, 0, 440, 0, 523, 0, 466, 0, 440, 0, 392, 0, 392, 0, 932, 880, 0, 932, 880, 0, 932, 392, 0, 392, 0, 932, 880, 0, 932, 880, 932, 0, 466, 466, 466, 466, 0, 587, 587, 587, 587, 523, 523, 523, 523, 698, 698, 698, 698, 0, 784, 784, 784, 784, 784, 784, 784, 784, 784, 784, 784, 784, 0, 523, 466, 440, 349, 392, 0, 392, 0, 587, 0, 523, 0, 466, 0, 440, 0, 440, 0, 440, 523, 0, 466, 0, 440, 0, 392, 0, 392, 0, 932, 880, 932, 880, 0, 932, 392, 0, 392, 932, 880, 932, 880, 932, 392, 0, 392, 587, 523, 0, 466, 0, 440, 0, 440, 0, 523, 0, 466, 0, 440, 0, 392, 0, 392, 932, 0, 880, 932, 0, 880, 0, 932, 392, 0, 392, 0, 932, 0, 880, 932, 0, 880, 932, 466, 466, 466, 466, 587, 587, 587, 587, 523, 523, 523, 523, 0, 698, 698, 698, 698, 0, 784, 784, 784, 784, 784, 784, 784, 784, 784, 784, 784, 784};
int noteDurations[267]={9000, 375, 281, 94, 281, 469, 375, 188, 281, 94, 281, 188, 375, 375, 281, 469, 375, 281, 375, 375, 375, 281, 281, 94, 375, 469, 94, 375, 94, 375, 375, 281, 375, 375, 375, 281, 375, 375, 375, 375, 281, 469, 375, 281, 375, 281, 562, 188, 375, 375, 469, 188, 375, 375, 94, 375, 281, 281, 94, 281, 94, 375, 281, 281, 94, 281, 375, 281, 94, 375, 469, 375, 188, 375, 281, 94, 375, 281, 94, 281, 469, 469, 188, 281, 375, 94, 469, 188, 375, 375, 375, 281, 469, 188, 469, 375, 375, 94, 281, 94, 375, 94, 375, 94, 281, 375, 94, 281, 281, 94, 562, 375, 94, 375, 94, 281, 281, 94, 281, 375, 562, 94, 375, 281, 375, 281, 94, 281, 375, 281, 281, 375, 375, 375, 375, 281, 281, 375, 281, 94, 375, 375, 375, 375, 281, 281, 281, 375, 375, 375, 375, 375, 94, 375, 375, 375, 281, 375, 375, 281, 94, 281, 94, 375, 281, 375, 375, 469, 281, 281, 94, 281, 375, 375, 281, 94, 281, 94, 375, 281, 281, 94, 281, 375, 375, 281, 94, 375, 375, 281, 375, 375, 281, 375, 375, 469, 375, 281, 281, 375, 469, 281, 375, 281, 469, 281, 469, 281, 375, 281, 375, 94, 281, 94, 375, 188, 375, 375, 94, 281, 281, 94, 281, 94, 469, 375, 94, 281, 188, 281, 94, 281, 375, 94, 281, 375, 375, 375, 375, 375, 281, 375, 281, 375, 375, 375, 281, 375, 94, 375, 375, 281, 375, 94, 375, 281, 375, 281, 375, 375, 281, 281, 281, 375, 281};
void setup() {
}
void loop(){ for(int thisNote = 0; thisNote < sizeof(melody)/4; thisNote ++){tone(8,melody[thisNote],noteDurations[thisNote]);delay(noteDurations[thisNote]);noTone(8);}}

 经过实际测试,该c++程序在ArduinoUno开发板上运行良好,蜂鸣器可以正确播放整首抬棺曲~



  1. 脉冲宽度调制(PWM)是一种模拟控制方式,根据相应载荷的变化来调制晶体管基极或MOS管栅极的偏置,来实现晶体管或MOS管导通时间的改变,从而实现开关稳压电源输出的改变。这种方式能使电源的输出电压在工作条件变化时保持恒定,是利用微处理器的数字信号对模拟电路进行控制的一种非常有效的技术。 ↩︎

这篇关于[Arduino Uno 实验]使用Arduino Uno开发板与无源蜂鸣器播放乐曲的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/607447

相关文章

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

Ubuntu如何分配​​未使用的空间

《Ubuntu如何分配​​未使用的空间》Ubuntu磁盘空间不足,实际未分配空间8.2G因LVM卷组名称格式差异(双破折号误写)导致无法扩展,确认正确卷组名后,使用lvextend和resize2fs... 目录1:原因2:操作3:报错5:解决问题:确认卷组名称​6:再次操作7:验证扩展是否成功8:问题已解

Qt使用QSqlDatabase连接MySQL实现增删改查功能

《Qt使用QSqlDatabase连接MySQL实现增删改查功能》这篇文章主要为大家详细介绍了Qt如何使用QSqlDatabase连接MySQL实现增删改查功能,文中的示例代码讲解详细,感兴趣的小伙伴... 目录一、创建数据表二、连接mysql数据库三、封装成一个完整的轻量级 ORM 风格类3.1 表结构

使用Docker构建Python Flask程序的详细教程

《使用Docker构建PythonFlask程序的详细教程》在当今的软件开发领域,容器化技术正变得越来越流行,而Docker无疑是其中的佼佼者,本文我们就来聊聊如何使用Docker构建一个简单的Py... 目录引言一、准备工作二、创建 Flask 应用程序三、创建 dockerfile四、构建 Docker

Python使用vllm处理多模态数据的预处理技巧

《Python使用vllm处理多模态数据的预处理技巧》本文深入探讨了在Python环境下使用vLLM处理多模态数据的预处理技巧,我们将从基础概念出发,详细讲解文本、图像、音频等多模态数据的预处理方法,... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核