当前位置: 首页 > news >正文

用Arduino单片机制作一个简单的音乐播放器

Arduino单片机上有多个数字IO针脚,可以输出数字信号,用于驱动发声器件,从而让它发出想要的声音。蜂鸣器是一种常见的发声器件,通电后可以发出声音。因此,单片机可以通过数字输出控制蜂鸣器发出指定的声音。另外,Arduino支持串口的通信方式,可以从电脑上接收数据,根据收到的数据确定所需发的声音。本文说明用上述方式如何通过Arduino单片机实现一个简单的音乐播放器。

一、项目实现的工具

(一)无源蜂鸣器

蜂鸣器有分有源蜂鸣器和无源蜂鸣器两种。其中,无源蜂鸣器内部不带震荡源,要通过震荡的信号来令其发出声音。音调随震荡频率的不同而不同[1]。

(二)Arduino单片机

Arduino单片机通过串口从电脑端获取音乐信息,然后根据音乐信息控制无源蜂鸣器的震荡频率,产生音乐效果。

(三)可调节电阻

用于控制无源蜂鸣器的电压大小,从而调节音量。

(四)树莓派电脑,运行基于Debian Linux的Raspberry Pi OS

一方面,编写Arduino程序并将其烧录进单片机;另一方面,将存储的音乐信息通过串口发送给Arduino。

(五)信号线

主要用于通电和传输信号

二、电路及程序设计

(一)电路

在该项目中,使用Arduino单片机的第7个IO针脚作为向蜂鸣器输出信号的针脚。注意Arduino单片机的输出是推挽输出,即高电平输出。

无源蜂鸣器有三个针脚,即电源(VCC),接地(GND)以及信号(I/O)。蜂鸣器在接通电源时,其随信号的高电平输入进行震荡,发出声音。而电源的电压决定了发声的响度。

电路大致如下

所以通过可调电阻,控制蜂鸣器的输入电压大小,从而调节音量大小。

(二)程序

在介绍程序之前,先简单介绍一些音乐的基本知识。我们通常的音乐是用八度音阶表示的。任何一个音符,和高八度的音相比,其音调(发声体震荡频率)相差两倍。每个八度区里有12个半音,所以每个半音的音调相差\sqrt[12]{2}倍。也就是说,如果简谱中的1的音调是xHz,那么{\stackrel{7}{\cdot}}的音调是x\over \sqrt[12]{2}Hz,而{}^\#1的音调是x*\sqrt[12]{2}2的音调是x*\sqrt[6]{2}。详细说明,见[2]。

本项目在Arduino程序中,用一个字节作为一个音符的信号。规定当该字节值为0x80时,代表的音调为1024Hz,可以理解为简谱中1的音调是1024Hz,然后字节值每相差1,就代表相差一个半音。因此,0x7F代表1024\over \sqrt[12]{2}Hz,即{\stackrel{7}{\cdot}};0x81代表1024*\sqrt[12]{2}Hz,即{}^\#1

另外,把0x00作为休止符。这里规定,每一个音符的时长为半秒。

在此,贴上Arduino里的C++程序

/*
Let's define the hex of the tonedefine a do: 1024Hz
0x80: do3
0x81: #do3
0x82: re3
0x7F: si2and each byte is for a note of 0.5s. So if you need do3 for 1s, then need 2 0x80*/const int buzzerPin = 7;
byte baseNote = 0x80;
int baseTone = 1024;
byte note;
void setup() {// put your setup code here, to run once:pinMode(buzzerPin, OUTPUT);Serial.begin(9600);
}void loop() {// put your main code here, to run repeatedly:if ((Serial.available() > 0) && (note = Serial.read()) && (note != 0x00)){Serial.print('Sing:');Serial.print(note);Serial.print(' ');//Now convert it into frequency of the toneint toneFreq;if (baseNote > note){ //received note is under base noteint noteDiff = baseNote - note;toneFreq = (int)((float)baseTone / (pow(2.0, noteDiff/12.0))); //between do and #do is 2^(1/12)}else{//received note is above base noteint noteDiff = note - baseNote;toneFreq = (int)((float)baseTone * (pow(2.0, noteDiff/12.0)));}Serial.print(toneFreq);Serial.print(' ');tone(buzzerPin, toneFreq);}else{noTone(buzzerPin);Serial.print('0');}delay(500);}

程序中,tone(buzzerPin, toneFreq)表式在buzzerPin数字输出中产生频率为toneFreqHz的震荡;而noTone(buzzerPin)表示让数字输出停止震荡。

因此,程序通过串口接收音符字节信息,然后根据该信息激活数字输出,让无源蜂鸣器发出音乐。

另外,在树莓派电脑中,运行一个python程序,通过读取指定格式的乐谱,产生音符数据,并将其通过串口输入到Arduino中。这里,先对乐谱的格式做一个规定。

这里,令乐谱为一个txt文件,该txt文件里每一个音符用一个1-3个字符的字符串组成,而音符和音符之间用空格分开。对于每一个音符,第一个字符是1-7之间的数字,意义和简谱的数字部分一样;第二的字符表明该音符在哪一个八度上,标准的八度号为3,所以如果没有该字符就默认为3处理;第三个字符可以不存在,也可以是+或-,表明是升调\#还是降调\flat。所以,假定乐谱文件如下:

13 33 23 52 0 43+ 63-

则对应的简谱是

1323 {\stackrel{5}{\cdot}} 0^\sharp 4^\flat 6

所以,把乐谱txt文件转换为Arduino读取的音乐数据的python程序如下

"""
This file gives the function to get music from file
base 0x80 is do3(13)
so do2 is 12 semitones lower than do3, do3+ is 1 semitone higher than do3, do3- is 1 semitone lower than do3
"""def loadMusicFile(filename):with open(filename,'r') as file:content = file.read()return content.split(' ')def convertFromMusicToArduino(filename):notes = loadMusicFile(filename)print(notes)musicData = []for n in notes:if len(n) == 0:continue;dataThis = 0x80if n[0] == '1':dataThis += 0elif n[0] == '2':dataThis += 2elif n[0] == '3':dataThis += 4elif n[0] == '4':dataThis += 5elif n[0] == '5':dataThis += 7elif n[0] == '6':dataThis += 9elif n[0] == '7':dataThis += 11else:dataThis = 0x00if len(n) > 1 and dataThis != 0x00:dataThis += (int(n[1]) - 3) * 0x0cif len(n) > 2 and dataThis != 0x00:if n[2] == '+':dataThis += 0x01elif n[2] == '-':dataThis -= 0x01musicData.append(dataThis)return musicData

注意0x0c表明十进制里的12。

然后,运行以下python程序,把音频文件转成Arduino支持的格式后,通过串口传入Arduino。python的串口连接的方式见[3]。

import serial
import os
from time import sleep
from getmusicFromFile import convertFromMusicToArduinoserialportName = '/dev/ttyACM1'
bps = 9600ser = serial.Serial(serialportName, int(bps), timeout=0.5, parity=serial.PARITY_NONE, stopbits=1)data = convertFromMusicToArduino('song.txt')data_bytes = bytes(data)
print(data_bytes)if (ser.isOpen()):print("Serial opened")sleep(1.8)ser.write(data_bytes)sleep(10)print(ser.read(30))ser.close()
print("done")

注意,在ser.isOpen()后,在传输数据前有一个sleep(1.8)。这个等待,是要让串口准备好后再传输数据,否则会传输失败,Arduino有可能未能收到数据。

三、实验效果

该实验,用歌曲"友谊地久天长"的第一小段作为乐谱,文件song.txt如下

52 52 13 13 0 13 13 0 33 33 23 23 23 13 23 0 33 33 13 13 0 13 33 33 53 53 63 63 63 0 0 0 63 63 53 53 53 33 33 33 13 13 23 23 23 13 23 23 33 23 13 13 13 62 62 62 52 52 13 13 13

运行效果见视频,播放的正是"友谊地久天长"的第一段。注意当我转动可调电阻时音量的变化。

友谊地久天长

参考资料

[1]有源蜂鸣器与无源蜂鸣器的驱动方式详解(精华版)_有源蜂鸣器和无源蜂鸣器的电路图-CSDN博客

[2]八度音阶和频率的关系_音阶与频率的关系-CSDN博客

[3]​​​​​​​用 Python 玩转串口(基于 pySerial)_python打开串口-CSDN博客

相关文章:

用Arduino单片机制作一个简单的音乐播放器

Arduino单片机上有多个数字IO针脚,可以输出数字信号,用于驱动发声器件,从而让它发出想要的声音。蜂鸣器是一种常见的发声器件,通电后可以发出声音。因此,单片机可以通过数字输出控制蜂鸣器发出指定的声音。另外&#x…...

软件工程相关

1.软件过程模型(重要) 1.1.瀑布模型 只适合需求明确的项目严格串行化,很长时间才能看到结果。严格区分阶段,每个阶段因果紧密相连,且要求每个阶段一次性解决该阶段的任务 1.2.原型模型(构造简易模型确定…...

速盾:游戏加速下载可以用cdn吗?

随着互联网的快速发展,游戏下载已经成为许多游戏玩家的常见需求。然而,由于游戏文件体积庞大,下载速度经常成为制约因素之一。为了解决这个问题,许多玩家开始寻找可以加速游戏下载速度的方法。其中一种常见的方法是使用CDN&#x…...

每日新闻掌握【2024年9月25日 星期三】

2024年9月25日 星期三 农历八月廿三 大公司/大事件 央行降低存量房贷利率,二套房贷最低首付比例下调到15% 国务院新闻办公室9月24日上午举行新闻发布会,中国人民银行、金融监管总局、中国证监会主要负责人介绍了金融支持经济高质量发展有关情况。多项重…...

8. Bug 与 Error

计算机程序中的缺陷通常被称为 bug。把它们想象成偶然爬进我们工作中的小东西,会让程序员感觉良好。当然,实际上是我们自己把它们放进去的。 如果程序是思想的结晶,我们可以将错误大致分为思想混乱造成的错误和将思想转化为代码时引入错误造成…...

论文 | Model-tuning Via Prompts Makes NLP Models Adversarially Robust

这篇论文研究了使用提示 (Prompting) 方法微调预训练语言模型,以提高其在对抗样本攻击下的鲁棒性。论文的主要贡献如下: 1.MVP 比 MLP-FT 更鲁棒: 论文比较了 MVP (Model-tuning Via Prompts) 和传统的 MLP-FT (Fine-tuning with an MLP head…...

828华为云征文|华为云Flexus云服务器X实例部署 即时通讯IM聊天交友软件——高性能服务器实现120W并发连接

营运版的即时通讯IM聊天交友系统:特点可发红包,可添加多条链接到用户网站和应用,安卓苹果APPPC端H5四合一 后端开发语言:PHP, 前端开发语言:uniapp混合开发。 集安卓苹果APPPC端H5四合一APP源码&#xff0…...

超好用的element的el-pagination分页组件二次封装-附源码及讲解

前言:在很多后台管理系统开发时总会有很多分页组件的使用,如果我们每次都用elementui官网的el-pagination去写的话,调整所有分页的样式就会很麻烦,而且页面内容也会很累赘繁琐。 讲解一个我经常使用的二次封装el-pagination组件&…...

【AIGC】通过OpenAi Canvas修改论文(附40条论文优化指令)

目录 1、用ChatGPT优化论文大纲和逻辑2、用ChatGPT充实论文内容3、用ChatGPT寻找案例和数据4、用ChatGPT检查语法和字词错误5、如何直接使用ChatGPT4o、o1、OpenAI Canvas6、OpenAI Canvas增强了啥?7、编程功能增强 在刚开始撰写学术论文时,很多小伙伴感…...

Kubernetes Pod详解

目录 1. Pod 介绍 1.1 Pod 结构 1.2 Pod 定义 2. Pod 配置 2.1 基本配置 2.2 镜像拉取 2.3 启动命令 2.4 环境变量 2.5 端口设置 2.6 资源配额 3. Pod 生命周期 3.1 创建和终止 3.2 初始化容器 3.3 钩子函数 3.4 容器探测 3.5 重启策略 4. Pod 调度 4.1 定向调…...

Vue2电商项目(七)、订单与支付

文章目录 一、交易业务Trade1. 获取用户地址2. 获取订单信息 二、提交订单三、支付1. 获取支付信息2. 支付页面--ElementUI(1) 引入Element UI(2) 弹框支付的业务逻辑(这个逻辑其实没那么全)(3) 支付逻辑知识点小总结 四、个人中心1. 搭建二级路由2. 展示动态数据(1). 接口(2).…...

你知道U盘怎么加密吗?

1、使用Windows BitLocker: 适用于Windows 10/11专业版及以上版本。 插入U盘,右键点击U盘图标,选择“启用BitLocker”。 设置密码,并选择加密选项,点击“开始加密”。 2、使用Mac的Disk Utility: 适用…...

【软件教程OBS下载使用】一篇文章教会你如何下载安装使用OBS-Studio

OBS Studio是全新的OBS(Open Broadcaster Software),是一款广泛应用的视频直播录制软件,跟经典版的区别就是,音频分路简单,在不出错的情况下性能优于经典版。可以说是高级版,目前仍然处于初期阶段,比起经典…...

鸿蒙next开发第一课03.ArkTs语法介绍-案例

前面已经学习了ArkTs的基本语法和DevEcoStudio的基本操作,接下来按照官方提示开发一个基本案例。 该案例是系统自带的demo,下载下来源代码后可以直接运行。 接下来我来演示如何运行demo。我在demo中加入了自己的注释。 切记:文件夹不能有中…...

HTML网页制作——设计系学生静态HTML网页设计作品

HTML网页制作——设计系学生静态HTML网页设计作品 网站主题为荷兰风格派,主要介绍荷兰风格设计的网站,由设计系学生亲自设计,独立开发网页,适用于学生自己的作品。 网站效果视频: 荷兰风格派(设计系学生网…...

智能翻译新纪元:4款英汉互译在线工具解析

大家好,我是一个喜欢找各种办公软件的人,今天咱们来聊聊那些让我们在英汉互译世界里如鱼得水的神器——福昕翻译在线、福昕翻译大师、海鲸AI论文翻译,还有DeepL翻译。这些家伙,简直就是我们跨语言交流的超级英雄! 1、…...

Cisco Meraki平台中国区注册

登陆下面网址注册cisco meraki中国区云平台账户 https://n4.meraki.cn/ 点击创建一个新账户 地区选择“china” 填写邮箱,名字,秘密,公司名称等信息,点击注册新账户 注册的邮箱会收到一封确认此邮箱的邮件,点击…...

分享国产RISC-V单片机通用

开源已经成为构建新技术生态的主流趋势。基于开源指令集 RISC-V 的软硬件生态正在飞速扩增,并且已经迅速扩展至个人 PC、服务器和人工智能等领域。RISC-V 的灵活性和可扩展性使其能够在应用处理器和AI加速领域迅速发展。 RAMSUN提供的RISC-V单片机,开源…...

java 网络知识 + 多线程问题

服务器: package p1007;import java.io.*; import java.net.*; import java.util.Random;public class Server {public static void main(String[] args) {int port 12345; // 服务端口try (ServerSocket serverSocket new ServerSocket(port)) {System.out.print…...

android 菜单不显示auto time zone菜单

packages\apps\Settings\res\xml\date_time_prefs.xml 有对应的xml packages\apps\Settings\src\com\android\settings\datetime\AutoTimeZonePreferenceController.java Overridepublic boolean isAvailable() {if (mIsFromSUW) {return false;}TimeZoneCapabilities time…...

【网络】每天掌握一个Linux命令 - iftop

在Linux系统中,iftop是网络管理的得力助手,能实时监控网络流量、连接情况等,帮助排查网络异常。接下来从多方面详细介绍它。 目录 【网络】每天掌握一个Linux命令 - iftop工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景…...

设计模式和设计原则回顾

设计模式和设计原则回顾 23种设计模式是设计原则的完美体现,设计原则设计原则是设计模式的理论基石, 设计模式 在经典的设计模式分类中(如《设计模式:可复用面向对象软件的基础》一书中),总共有23种设计模式,分为三大类: 一、创建型模式(5种) 1. 单例模式(Sing…...

Appium+python自动化(十六)- ADB命令

简介 Android 调试桥(adb)是多种用途的工具,该工具可以帮助你你管理设备或模拟器 的状态。 adb ( Android Debug Bridge)是一个通用命令行工具,其允许您与模拟器实例或连接的 Android 设备进行通信。它可为各种设备操作提供便利,如安装和调试…...

相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解

【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了: 这一篇我们开始讲: 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下: 一、场景操作步骤 操作步…...

【SpringBoot】100、SpringBoot中使用自定义注解+AOP实现参数自动解密

在实际项目中,用户注册、登录、修改密码等操作,都涉及到参数传输安全问题。所以我们需要在前端对账户、密码等敏感信息加密传输,在后端接收到数据后能自动解密。 1、引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId...

Vue2 第一节_Vue2上手_插值表达式{{}}_访问数据和修改数据_Vue开发者工具

文章目录 1.Vue2上手-如何创建一个Vue实例,进行初始化渲染2. 插值表达式{{}}3. 访问数据和修改数据4. vue响应式5. Vue开发者工具--方便调试 1.Vue2上手-如何创建一个Vue实例,进行初始化渲染 准备容器引包创建Vue实例 new Vue()指定配置项 ->渲染数据 准备一个容器,例如: …...

基于数字孪生的水厂可视化平台建设:架构与实践

分享大纲&#xff1a; 1、数字孪生水厂可视化平台建设背景 2、数字孪生水厂可视化平台建设架构 3、数字孪生水厂可视化平台建设成效 近几年&#xff0c;数字孪生水厂的建设开展的如火如荼。作为提升水厂管理效率、优化资源的调度手段&#xff0c;基于数字孪生的水厂可视化平台的…...

HBuilderX安装(uni-app和小程序开发)

下载HBuilderX 访问官方网站&#xff1a;https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本&#xff1a; Windows版&#xff08;推荐下载标准版&#xff09; Windows系统安装步骤 运行安装程序&#xff1a; 双击下载的.exe安装文件 如果出现安全提示&…...

【Zephyr 系列 10】实战项目:打造一个蓝牙传感器终端 + 网关系统(完整架构与全栈实现)

🧠关键词:Zephyr、BLE、终端、网关、广播、连接、传感器、数据采集、低功耗、系统集成 📌目标读者:希望基于 Zephyr 构建 BLE 系统架构、实现终端与网关协作、具备产品交付能力的开发者 📊篇幅字数:约 5200 字 ✨ 项目总览 在物联网实际项目中,**“终端 + 网关”**是…...

UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)

UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中&#xff0c;UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化&#xf…...