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

STM32F103C8T6驱动无源蜂鸣器播放《两只老虎》完整教程(附源码)

STM32F103C8T6驱动无源蜂鸣器播放《两只老虎》完整教程附源码蜂鸣器作为嵌入式开发中最基础的外设之一常被用于系统报警、状态提示等场景。但你是否想过通过精确控制PWM频率和节奏可以让这个简单的元件演奏出熟悉的旋律本文将带你用STM32的TIM定时器实现无源蜂鸣器播放《两只老虎》的完整过程。1. 硬件准备与原理分析1.1 无源蜂鸣器工作特性无源蜂鸣器与有源蜂鸣器的核心区别在于驱动方式有源蜂鸣器内置振荡电路只需提供直流电压即可发声无源蜂鸣器需要外部提供方波信号才能工作音乐播放场景必须使用无源蜂鸣器因为音高由驱动频率决定C4261HzD4294Hz等音长由信号持续时间控制可通过PWM精确调节波形特性1.2 STM32的PWM生成机制STM32F103C8T6通过TIM定时器产生PWM信号的关键配置参数参数作用计算公式ARR (Auto-reload)决定PWM周期频率 定时器时钟/(PSC1)/(ARR1)PSC (Prescaler)时钟预分频系数实际时钟 72MHz/(PSC1)CCR (Capture Compare)决定占空比占空比 CCR/(ARR1)提示音乐播放时通常固定50%占空比重点调节ARR值改变频率2. 音乐编程基础2.1 音阶频率对照表《两只老虎》主要使用中音区(C4-B4)的七个基本音阶音符频率(Hz)计算值(ARR) PSC71C4261.633830D4293.663412E4329.633040F4349.232870G4392.002558A4440.002279B4493.882032计算公式ARR (72000000 / (PSC1)) / frequency - 1 (72000000 / 72) / frequency - 1 1000000 / frequency - 12.2 节拍时间控制四四拍歌曲的典型节拍时长BPM120时节拍类型持续时间(ms)全音符2000二分音符1000四分音符500八分音符250《两只老虎》简谱对应的节拍序列C4(1) D4(1) E4(1) C4(1) | C4(1) D4(1) E4(1) C4(1) | E4(1) F4(1) G4(2) | E4(1) F4(1) G4(2) | G4(0.5) A4(0.5) G4(0.5) F4(0.5) E4(1) C4(1) | G4(0.5) A4(0.5) G4(0.5) F4(0.5) E4(1) C4(1) | C4(1) G3(1) C4(2) | C4(1) G3(1) C4(2) |3. 工程实现步骤3.1 硬件连接使用STM32F103C8T6最小系统板与无源蜂鸣器模块连接蜂鸣器 → PA0 (TIM2_CH1) 蜂鸣器- → GND注意无源蜂鸣器没有极性但需串联100Ω限流电阻3.2 代码实现3.2.1 PWM初始化配置// pwm.h #define BUZZER_TIM TIM2 #define BUZZER_CHANNEL TIM_CHANNEL_1 void PWM_Init(void); void PWM_SetFreq(uint16_t freq); void PWM_PlayNote(uint16_t freq, uint32_t duration);// pwm.c #include stm32f10x.h void PWM_Init(void) { GPIO_InitTypeDef GPIO_InitStruct; TIM_TimeBaseInitTypeDef TIM_InitStruct; TIM_OCInitTypeDef TIM_OCInitStruct; // 1. 开启时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); // 2. 配置GPIO GPIO_InitStruct.GPIO_Pin GPIO_Pin_0; GPIO_InitStruct.GPIO_Mode GPIO_Mode_AF_PP; GPIO_InitStruct.GPIO_Speed GPIO_Speed_50MHz; GPIO_Init(GPIOA, GPIO_InitStruct); // 3. 配置定时器基础 TIM_InitStruct.TIM_Period 999; // 初始10kHz TIM_InitStruct.TIM_Prescaler 71; // 72MHz/721MHz TIM_InitStruct.TIM_ClockDivision TIM_CKD_DIV1; TIM_InitStruct.TIM_CounterMode TIM_CounterMode_Up; TIM_TimeBaseInit(BUZZER_TIM, TIM_InitStruct); // 4. 配置PWM输出 TIM_OCInitStruct.TIM_OCMode TIM_OCMode_PWM1; TIM_OCInitStruct.TIM_OutputState TIM_OutputState_Enable; TIM_OCInitStruct.TIM_OCPolarity TIM_OCPolarity_High; TIM_OCInitStruct.TIM_Pulse 500; // 50%占空比 TIM_OC1Init(BUZZER_TIM, TIM_OCInitStruct); TIM_Cmd(BUZZER_TIM, ENABLE); } void PWM_SetFreq(uint16_t freq) { uint16_t arr (uint16_t)(1000000 / freq) - 1; TIM_SetAutoreload(BUZZER_TIM, arr); TIM_SetCompare1(BUZZER_TIM, arr / 2); // 保持50%占空比 }3.2.2 音乐播放逻辑// music.h typedef struct { uint16_t freq; uint16_t duration; } Note; #define TEMPO 120 // BPM #define QUARTER_NOTE (60000 / TEMPO) extern const Note song[]; extern const uint16_t song_length;// music.c #include music.h const Note song[] { {262, QUARTER_NOTE}, // C4 {294, QUARTER_NOTE}, // D4 {330, QUARTER_NOTE}, // E4 {262, QUARTER_NOTE}, // C4 // ... 完整乐谱 }; const uint16_t song_length sizeof(song)/sizeof(Note);3.2.3 主程序// main.c #include stm32f10x.h #include pwm.h #include music.h #include delay.h int main(void) { Delay_Init(); PWM_Init(); while(1) { for(int i0; isong_length; i) { PWM_SetFreq(song[i].freq); Delay_ms(song[i].duration); } PWM_SetFreq(0); // 停止发声 Delay_ms(2000); // 间隔2秒 } }4. 进阶优化技巧4.1 节拍精度提升使用SysTick定时器替代Delay_ms实现更精确的节奏控制void SysTick_Handler(void) { static uint32_t counter 0; if(counter note_duration) { counter 0; play_next_note(); } }4.2 多任务处理在RTOS环境中创建独立播放任务void buzzer_task(void *params) { while(1) { play_song(two_tigers); vTaskDelay(pdMS_TO_TICKS(5000)); } } xTaskCreate(buzzer_task, Buzzer, 128, NULL, 2, NULL);4.3 音效增强通过调制PWM占空比实现音色变化void PWM_SetEnvelope(uint16_t freq, uint16_t duration) { // 淡入效果 for(int i1; i10; i) { TIM_SetCompare1(BUZZER_TIM, (ARR/i)); Delay_ms(duration/20); } // 淡出效果 for(int i10; i1; i--) { TIM_SetCompare1(BUZZER_TIM, (ARR/i)); Delay_ms(duration/20); } }

相关文章:

STM32F103C8T6驱动无源蜂鸣器播放《两只老虎》完整教程(附源码)

STM32F103C8T6驱动无源蜂鸣器播放《两只老虎》完整教程(附源码) 蜂鸣器作为嵌入式开发中最基础的外设之一,常被用于系统报警、状态提示等场景。但你是否想过,通过精确控制PWM频率和节奏,可以让这个简单的元件演奏出熟悉…...

STM32F429+LAN8720A网络实战:CubeMX一键配置LWIP+FreeRTOS,从原理图到Ping通全流程避坑

STM32F429与LAN8720A网络开发实战:从硬件连接到LWIP调通的深度解析 在嵌入式系统开发中,网络功能的集成往往是项目从原型走向实际应用的关键一步。STM32F429系列微控制器凭借其强大的性能和丰富的外设资源,成为许多工业级应用的理想选择。而L…...

从零部署到QPS 12,800:EF Core 10 + Azure AI Search向量管道搭建,附可审计的迁移Checklist

第一章:EF Core 10 向量搜索扩展的演进与定位EF Core 10 并未原生内置向量搜索能力,但其可扩展性架构为第三方向量搜索集成提供了坚实基础。随着 AI 应用对语义检索需求激增,社区与厂商开始围绕 EF Core 构建轻量、数据库感知的向量搜索扩展—…...

避坑指南:为什么你的Kalibr双目+IMU标定总失败?从参数配置到数据采集全解析

Kalibr双目IMU标定实战避坑指南:从参数陷阱到数据采集的完整解决方案 当你在机器人或自动驾驶项目中尝试进行传感器融合时,双目相机与IMU的联合标定往往是第一个技术门槛。许多工程师在初次使用Kalibr工具链时会遇到各种令人沮丧的报错——优化失败、角点…...

Anthropic 官方技能最佳实践:14 个可复用的 Agent Skills 设计模式

在 Agent Skills 的生态中,技能大致可以分为两类。 一类是任务型技能(通常设置 disable-model-invocation: true),对应一整套步骤化流程,比如部署、提交或安全审查,用户一般通过 /skill-name 直接触发。 另…...

告别盲调!用万用表和GD32 DAC玩点真的:生成精准电压信号测试你的电路

告别盲调!用万用表和GD32 DAC玩点真的:生成精准电压信号测试你的电路 在电子设计的世界里,调试电路就像医生诊断病人——没有准确的测量工具,再精妙的电路设计也可能功亏一篑。而GD32的DAC(数字模拟转换器&#xff09…...

从零到可视化:手把手教你用RocketMQ Console在Windows上搭建消息队列监控面板

从零到可视化:手把手教你用RocketMQ Console在Windows上搭建消息队列监控面板 在分布式系统架构中,消息队列作为解耦和异步通信的核心组件,其运行状态的实时监控至关重要。RocketMQ Console作为官方提供的可视化工具,能将晦涩的命…...

从Slab到内存池:深入拆解Linux内核如何高效管理‘碎片化’小内存(以task_struct为例)

从Slab到内存池:深入拆解Linux内核如何高效管理‘碎片化’小内存(以task_struct为例) 在操作系统内核的开发中,内存管理一直是性能优化的核心战场。尤其对于像task_struct这样频繁创建和销毁的小内存对象,传统的内存分…...

STM32 串口通信 (UART) 全栈底层复习指南

目录 一、 物理层与通信协议基础 (底层时序) 1. 硬件连接规则 2. 通信时序与数据帧 (以最常用的 10 位标准帧 8N1 为例) 二、 UART 底层硬件架构 (双缓冲机制) 1. 接收双缓冲:移位寄存器 & RDR (接收数据寄存器) 2. 发送双缓冲:TDR (发送数据寄…...

从一次真实的渗透测试说起:我是如何通过SQL注入拿下BeeCMS 4.0后台并上传Webshell的

实战剖析:BeeCMS 4.0安全漏洞链的完整利用路径 当阳光透过百叶窗在键盘上投下斑驳光影时,我正在对某企业官网进行常规渗透测试。这个使用BeeCMS 4.0搭建的网站看似普通,却意外成为了展示经典漏洞链的绝佳案例。本文将完整还原从发现漏洞到获取…...

苹果权力交接落定,John Ternus接棒库克,三大难题待解

John Ternus接棒库克,苹果权力重心转移 周一,苹果宣布硬件工程高级副总裁John Ternus将于9月1日接替库克出任首席执行官,库克则转任执行董事长,继续负责苹果与全球政策制定者的关系维护。此次权力交接备受瞩目,Ternus从…...

vben开发入门1:创建和运行项目

了解官网 说明:企业级管理系统框架,开箱即用,简单高效 官网地址: https://doc.vben.pro/ 源码地址:https://github.com/vbenjs/vue-vben-admin 演示地址: https://www.vben.pro/ https://ant.vben.pro/ ht…...

告别RTC日期混乱:用STM32CubeMX和HAL库实现可靠的时间戳方案

告别RTC日期混乱:用STM32CubeMX和HAL库实现可靠的时间戳方案 在工业控制和通信设备开发中,精确可靠的时间管理往往是系统稳定性的关键。许多开发者在使用STM32的RTC模块时都遇到过这样的困扰:设备断电重启后,日期信息丢失或错误&a…...

如何回收未使用的区_DEALLOCATE UNUSED释放高水位上空间

DEALLOCATE UNUSED 不释放HWM空间,因它仅回收段末尾完全未用的extents,不移动HWM;HWM下已格式化但空闲的块仍被锁定,需先执行SHRINK SPACE COMPACT下移HWM再配合使用。DEALLOCATE UNUSED 为什么没释放高水位线(HWM&…...

BetterJoy终极指南:3步让Switch控制器在PC上完美兼容XInput和模拟器

BetterJoy终极指南:3步让Switch控制器在PC上完美兼容XInput和模拟器 【免费下载链接】BetterJoy Allows the Nintendo Switch Pro Controller, Joycons and SNES controller to be used with CEMU, Citra, Dolphin, Yuzu and as generic XInput 项目地址: https:/…...

AI写论文实用攻略!4款AI论文生成工具,打造优质学术论文!

撰写学术论文的困难与 AI 工具的解决方案 撰写学术论文、毕业论文或职称论文时,很多学者会遇到各种困难。人工撰写论文常常像是大海捞针,海量的文献资料让人苦于寻找相关信息;而复杂的格式规范又把人搞得不知所措,严格的要求让人…...

如何用 Fullscreen API 监听全屏切换状态并调整界面 UI

可通过监听 fullscreenchange 事件并检查 document.fullscreenElement 来准确判断全屏状态,据此动态调整UI;全屏API须在用户手势中调用,退出时用 document.exitFullscreen() 并处理 Promise;CSS 可配合 :fullscreen 伪类和 class …...

定制开发 vs 模板小程序

一、模板小程序现成成品,一键开通、上线快、初期价格低。功能固定无法深度修改,界面、流程、逻辑不能自定义。多为按年付费、账号租用,不含源码,版权不归自己。同质化严重,同行界面一模一样,无品牌差异化。…...

Ubuntu 24.04 LTS 新特性与长期支持策略解析

1. Ubuntu 24.04 LTS "Noble Numbat" 深度解析作为一名长期跟踪Linux发行版演进的技术博主,我第一时间在物理机和虚拟机环境完成了Ubuntu 24.04 LTS的部署测试。这个代号为"Noble Numbat"(高贵袋食蚁兽)的版本确实带来了…...

Agent调用工具失败?5个常见Tool Registration错误及修复方案(2026 全新深度排查指南 全程避坑,亲测有效)

一、为什么 Agent 工具注册如此容易出错? 1.1 LangChain Agent 的工具调用机制 Agent 通过 工具注册表(Tool Registry) 管理可用工具: #mermaid-svg-eZJSPSLtCp2kav5W{font-family:"trebuchet ms",verdana,arial,sans…...

洲际油气一路暴跌解股,隆基绿能反复磨底,光伏行业何时迎来拐点

全局总结论 风险提示,再逐个拆解深成指、洲际油气、隆基绿能,把你遇到的指数牛市、个股暴跌、白马阴跌、反弹就被砸的底层逻辑全部讲透。⚠️ 风险提示:以下仅为市场基本面、资金面、行业逻辑分析,不构成任何投资建议、买卖指导&…...

免费开源的WPS AI插件 察元AI助手:getSelectedText 与 resolveDocumentInput 的组合使用

摘要本篇聚焦 documentActions 中选区与全文的衔接。实现新助手时,应明确 sourceMode,并在无选区时是否允许回退全文,以避免误处理整篇公文。关键词选区;全文;sourceMode扩展阅读与维护提示本篇围绕「getSelectedText 与 resolveDocumentInpu…...

别再死记硬背了!用PyTorch手把手带你理解ReLU和Sigmoid激活函数到底在干啥

激活函数可视化实验:用PyTorch解剖ReLU与Sigmoid的神经元行为 当你在PyTorch中第一次构建神经网络时,是否曾被激活函数的选择困扰过?为什么简单的ReLU能击败曾经风靡的Sigmoid?让我们通过三个维度来解构这个现象:数学特…...

AspectJ编译期织入实战

JDK动态代理对final类/方法增强无效,CGLIB因继承机制无法代理final类/方法。当业务场景中必须使用final类(如工具类、第三方依赖类)或final方法时,Spring AOP(动态代理)已无法满足需求,此时需使…...

线性判别分析LDA

一、降维的基础背景降维的概念与必要性:在机器学习中,降维是指在限定条件下减少随机变量的个数,以提取出不相关的主变量 。由于实际数据常面临多重共线性(导致模型泛化能力弱、高维空间稀疏难以找到特征等问题)&#x…...

每日一Go-55、分布式 ID 生成(雪花算法 / Segment / Redis / DB)

一、为什么分布式系统一定要“自己造ID”? 单机时代,利用数据库的自增ID AUTO_INCREMENT但是在微服务/多实例/分库分表的情况下,会出现:ID冲突数据迁移困难顺序失控跨库无法唯一定位二、分布式ID的核心指标 一个靠谱的ID方案&…...

别再手动对齐了!用Creo的骨架模型做装配,效率提升不止一点点

别再手动对齐了!用Creo的骨架模型重构你的装配设计流程 当你在设计一个包含二十个运动部件的机械臂时,突然接到客户修改行程参数的需求——传统装配方式下,这意味着要逐个调整每个零件的安装位置、重新计算配合间隙、反复检查干涉区域。这种&…...

从HMM到BiLSTM-CRF:我的NER模型进化之路与性能对比实验报告

从HMM到BiLSTM-CRF:我的NER模型进化之路与性能对比实验报告 三年前第一次接触命名实体识别(NER)任务时,我完全没想到这个看似简单的序列标注问题会让我在模型迭代的路上走这么远。从最初用HMM处理简单场景,到引入CRF解决标签依赖问题&#xf…...

从Simulink仿真到STM32烧录:手把手搭建SVPWM算法验证闭环(附模型和工程)

SVPWM算法在电机控制中的全流程实现:从Simulink仿真到STM32硬件验证 电机控制算法的开发往往需要在理论验证和硬件实现之间反复迭代。SVPWM(空间矢量脉宽调制)作为现代电机控制的核心技术,其实现过程涉及数学建模、仿真验证、代码…...

数百种蛋白同步解析:抗体芯片如何重塑WB技术边界

摘要:高通量Western Blot技术通过将传统蛋白质印迹实验与微阵列芯片平台相结合,实现了单次实验中对数百种蛋白质表达水平的同步检测。该技术以抗体芯片为核心载体,显著提升了实验通量与数据可重复性,在蛋白质组学研究中展现出重要…...