C++数字化声音信号处理和数控振荡合成

🎯要点

  1. C++声音数控方法及应用实例:🎯加法合成、减法合成、共振峰合成、粒状合成、调频 (FM) 合成、线性算术合成、相位失真合成、扫描合成、矢量合成、虚拟模拟合成、波表合成、物理建模。

  2. C++声音​数字​化实现​:🎯声音合成引擎作用:初始化、使用GUI控件、渲染音频。

    1. 🎯合成器三种基本操作模式:单声道、单声道齐奏和复调,区别音符关闭事件和音符结束信号。🖊拆分MIDI事件为MIDI音符事件和连续控制器事件。🖊设置语音对象的失谐、平移和相位偏移。🖊储存声音状态和MIDI事件。🖊复调操作:代码中声明合成器类型、目标设备、CPU 功率和其他信息的语音对象。🖊复调和声音时间标签。

    2. 🎯使用模块创建自定义C++音频模块核:🖊加载模块并调用自定义模块,🖊在代码中使用调制器,🖊访问调制器输出,🖊使用振荡器和访问音频输出:四振荡器虚拟模拟合成器、拨弦算法合成器、四振荡器波表合成器、脉冲编码调制振荡器,🖊使用滤波器对象和访问滤波器音频输入输出,🖊提供MIDI数据结构渲染音频,🖊GUI控件。

    3. 🎯调制计算:🖊调制源:低频振荡器、包络发生器、MIDI 连续控制器、力度和音符编号,🖊衰减计算:凹变换和凸变换,🖊计算MIDI 起音速度和衰减音数缩放,🖊恒定功率平移和交叉淡入淡出控制,🖊包络调制计算,🖊双极调制线性调频,🖊多个振荡器量化输出信号,🖊斜坡调制,滑音调制,🖊计算振荡器音高,🖊脉冲宽度调制,🖊振荡器与内部主振荡器的时基硬同步,🖊过滤键跟踪。

    4. 🎯包络发生器和数控放大器、🎯低频振荡器、🎯波表振荡器、🎯虚拟模拟 (VA) 振荡器、🎯脉冲编码调制样本回放振荡器。🎯合成器滤波、🎯弹拨弦模型、🎯调制矩阵、🎯波变形和序列。

🍇C++声音合成和音效处理

由于不同系统之间计算机扬声器发出的声音有很大差异,因此我们将首先编写一个 .wave 文件。波形文件头有很多可选部分,但我们只会关注完成工作所需的最低限度。我们的波形文件头结构如下所示:

//this struct is the minimal required header data for a wav file
struct SMinimalWaveFileHeader
{//the main chunkunsigned char m_szChunkID[4];uint32 m_nChunkSize;unsigned char m_szFormat[4];//sub chunk 1 "fmt "unsigned char m_szSubChunk1ID[4];uint32 m_nSubChunk1Size;uint16 m_nAudioFormat;uint16 m_nNumChannels;uint32 m_nSampleRate;uint32 m_nByteRate;uint16 m_nBlockAlign;uint16 m_nBitsPerSample;//sub chunk 2 "data"unsigned char m_szSubChunk2ID[4];uint32 m_nSubChunk2Size;//then comes the data!
};

以下是填充结构并将其写入磁盘的函数:

bool WriteWaveFile(const char *szFileName, void *pData, int32 nDataSize, int16 nNumChannels, int32 nSampleRate, int32 nBitsPerSample)
{//open the file if we canFILE *File = fopen(szFileName,"w+b");if(!File){return false;}SMinimalWaveFileHeader waveHeader;//fill out the main chunkmemcpy(waveHeader.m_szChunkID,"RIFF",4);waveHeader.m_nChunkSize = nDataSize + 36;memcpy(waveHeader.m_szFormat,"WAVE",4);//fill out sub chunk 1 "fmt "memcpy(waveHeader.m_szSubChunk1ID,"fmt ",4);waveHeader.m_nSubChunk1Size = 16;waveHeader.m_nAudioFormat = 1;waveHeader.m_nNumChannels = nNumChannels;waveHeader.m_nSampleRate = nSampleRate;waveHeader.m_nByteRate = nSampleRate * nNumChannels * nBitsPerSample / 8;waveHeader.m_nBlockAlign = nNumChannels * nBitsPerSample / 8;waveHeader.m_nBitsPerSample = nBitsPerSample;//fill out sub chunk 2 "data"memcpy(waveHeader.m_szSubChunk2ID,"data",4);waveHeader.m_nSubChunk2Size = nDataSize;//write the headerfwrite(&waveHeader,sizeof(SMinimalWaveFileHeader),1,File);//write the wave data itselffwrite(pData,nDataSize,1,File);//close the file and return successfclose(File);return true;
}

现在,我们将生成一些音频数据并制作一个真正的波形文件!由于它们很容易生成,我们将使用锯齿波作为声音。

int nSampleRate = 44100;
int nNumSeconds = 4;
int nNumChannels = 1;

采样率定义每秒有多少个音频数据样本。 音频数据流只不过是数字流,每个数字都是一个音频样本,因此采样率就是每秒有多少个数字音频数据。 使用的数字越少,声音文件的“水平分辨率”就越小,或者波形数据每秒振幅变化的次数就越少。

采样率还定义了可以存储在音频流中的最大频率。 您可以存储的最大频率是采样率的一半。 换句话说,在 44100 采样率的情况下,您可以存储的最大频率为 22,050hz。 人耳的最大可听频率约为 20,000hz,因此使用 44100 的采样率应该可以满足大多数需求(出于复杂的技术原因,您可能需要更高的采样率,但这对于现在来说已经足够了!)。

秒数是波持续的时间(以秒为单位),通道数是有多少个音频通道。由于这是单声道声音,因此只有一个音频通道。

int nNumSamples = nSampleRate * nNumChannels * nNumSeconds;
int32 *pData = new int32[nNumSamples];

这里我们计算有多少实际音频样本,然后分配空间来保存音频数据。 我们使用 32 位整数,但您也可以使用 16 位整数。 音频样本中的位数表示音频数据的垂直分辨率,或者有多少个唯一值。 在 16 位整数中,有 65536 个不同的值,在 32 位中,有 42 亿个不同的值。 如果您将数据视为图表上的图,每个样本的位数越多,采样率越高,您的图表就越接近真实的数据。 较少的位数和较低的采样率意味着它与您尝试建模的真实数据相距较远,这将导致音频听起来不太正确。

int32 nValue = 0;
for(int nIndex = 0; nIndex < nNumSamples; ++nIndex)
{nValue += 8000000;pData[nIndex] = nValue;
}

在这里,我们实际上正在创建我们的波浪数据。 我们利用这样一个事实:如果你有一个 int 接近可以存储的最大值,然后添加更多值,它将回绕到 int 可以存储的最小值。 如果你在图表上看它,它看起来像锯齿波,即我们正在创建锯齿波! 通常你不会以这种方式创建它们,因为我们这样做的方式对耳朵来说很刺耳,并且引入了一种称为混叠的东西。

您可以更改添加到 nValue 的数量来更改所得波的频率。 添加较小的数字使其频率较低,添加较大的数字使其频率较高。

WriteWaveFile("outmono.wav",pData,nNumSamples * sizeof(pData[0]),nNumChannels,nSampleRate,sizeof(pData[0])*8);
delete[] pData;

后,我们编写波形文件并释放内存。

写入立体声文件

立体声文件中唯一真正发生变化的是有 2 个通道而不是 1 个,并且我们生成音频数据的方式略有不同。 由于有 2 个通道,一个用于左通道,一个用于右通道,因此对于相同采样率和时间长度的波形文件,实际上存在双倍的音频数据,因为每个通道都需要一整套数据。

参阅一:计算思维
参阅二:亚图跨际

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

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

相关文章

Jmeter 性能压测-TPS与QPS

1、TPS和QPS的区别 TPS&#xff1a;意思是每秒事务数&#xff0c;具体事务的定义都是人为的&#xff0c;可以一个接口、多个接口、一个业务流程等等。 一个事务是指事务内第一个请求发送到接收到最后一个请求的响应的过程&#xff0c;以此来计算使用的时间和完成的事务个数。…

「PHP系列」PHP超级全局变量详解

文章目录 一、$GLOBALS1. 在函数内部访问全局变量2. 在函数内部修改全局变量3. 注意事项 二、$_SERVER三、$_REQUEST1. 从 GET 请求中获取数据2. 从 POST 请求中获取数据3. 注意事项 四、$_POST五、$_GET六、$_FILES七、$_ENV八、$_COOKIE九、$_SESSION十、相关链接 一、$GLOBA…

python UTF-8解码及脚本头的标注

在Python中,如果你需要将编码为UTF-8的字节串解码为Unicode字符串,你可以使用内置的str类型的decode方法,或者使用bytes.decode()方法。但通常情况下,如果你已经在Python 3中处理字符串,你可以直接将字节串(类型bytes)转换为字符串(类型str)。 例如: python # 假设…

S32K324 数据初始化Rom到Ram Copy的方式

文章目录 前言基础知识ld文件中的段定义ld文件中的符号定义 ld定义copy地址范围启动文件中的定义Copy的使用总结 前言 之前一直不理解在ld文件中加__xxx_ram_start,__xxx_rom_start,__xxx_rom_end这些的作用&#xff0c;也不清楚原理。前几天遇到一个内存copy的问题&#xff0…

从入门到放弃:Docker基础教程

一、引言 1. 什么是Docker Docker是一种用于开发、交付和运行应用程序的平台。它通过将应用程序打包成一个可以轻松部署的容器来实现隔离&#xff0c;从而简化了应用程序部署的流程。 2. Docker能解决什么问题 传统的应用程序部署和管理方式往往存在着各种问题&#xff0c;…

HarmonyOS时区和语言设置-使用相关api实现系统语言和地区设置

介绍 本示例展示了i18n&#xff0c;intl&#xff0c;resourceManager在eTS中的使用&#xff0c;使用相关api实现系统语言和地区设置、时间和时区设置&#xff0c;展示了区域格式化示例。 效果预览 使用说明 1.启动应用&#xff0c;进入应用&#xff0c;首页分为三个按钮&…

【JVM】GC导致的性能问题排查与解决方案,日志、堆分析工具介绍

一、必要性 重要应用程序在使用过程中&#xff0c;忽然无法响应用户请求&#xff0c;排查发现网络联通无问题&#xff0c;gateway能够正常接收分发请求&#xff0c;应用进程正常&#xff0c;正常向注册中心发送请求&#xff0c;但是接收http请求全部返回报错。 添加gc后发现内…

C#项目引用解决方案中其他项目dll时,出现黄色感叹号的解决方案

问题引入 今天拿着老师傅的老项目&#xff0c;需要做通讯调试&#xff0c;说测试一下&#xff0c;便添加了一个项目A来编写结构体&#xff0c;然后在窗体程序项目B中引用A&#xff0c;发现B一引用A&#xff0c;在B项目的引用下面A就多了个黄色感叹号&#xff0c;一编译B项目&am…

网工内推 | 上市公司网工,最高30K,思科认证优先,多次晋升机会

01 牧原股份 招聘岗位&#xff1a;网络工程师 职责描述&#xff1a; 1、负责公司及下属子公司办公网络及IOT网络架构规划、设计、重大网络变更评审或实施及重大疑难问题处理&#xff1b; 2、负责公司网络运维监控体系、自动化网络运维及服务体系&#xff0c;并持续优化改进&am…

20240408在全志H3平台的Nano Pi NEO CORE开发板的eMMC刷Ubuntu Core 16.04

20240408在全志H3平台的Nano Pi NEO CORE开发板的eMMC刷Ubuntu Core 16.04 2024/4/8 20:46 参考资料&#xff1a; https://wiki.friendlyelec.com/wiki/index.php/NanoPi_NEO_Core/zh#.E5.AE.89.E8.A3.85.E7.B3.BB.E7.BB.9F [ OK ] Created slice Slice /system/getty. [ …

arm 的CoreLink 是什么?

ARM的CoreLink是一套由ARM公司开发的系统互连IP解决方案&#xff0c;旨在为片上系统&#xff08;SoC&#xff09;提供高性能和高效率的互连架构。CoreLink系列包括多种技术和组件&#xff0c;每个都针对特定的系统设计需求进行了优化。以下是CoreLink系列的一些关键组件及其使用…

代码随想录Day48

Day 48 动态规划part09 今日任务 198.打家劫舍213.打家劫舍II337.打家劫舍III 代码实现 基础打家劫舍 class Solution {public static int rob(int[] nums) {if (nums null || nums.length 0) return 0;if (nums.length 1) return nums[0];int[] dp new int[nums.leng…

获取淘宝销量API商品详情页原数据APP接口:测试key获取(含测试链接)

淘宝/天猫获得淘宝app商品详情原数据 API 返回值说明 item_get_app-获得淘宝app商品详情原数据 公共参数 名称类型必须描述keyString是调用key&#xff08;必须以GET方式拼接在URL中&#xff09;secretString是调用密钥api_nameString是API接口名称&#xff08;包括在请求地…

注入类型(一)

一、整数类型 1 and 11 # 1 or 11 -- 二、字符串类型 1 and 11 -- 1 and 11 # 三、搜素类型 搜索型注入&#xff0c;需要额外的考虑后面的问题 总结&#xff1a; 前闭合 " ) ") % %" 无 后闭合 " ( (" % …

百度松果菁英班——机器学习实践五:明星图片爬取

飞桨AI Studio星河社区-人工智能学习与实训社区 &#x1f96a;图片爬取 import requests import os import urllib ​ class GetImage():def __init__(self,keyword大雁,paginator1):# self.url: 链接头self.url http://image.baidu.com/search/acjson?self.headers {User…

Linux安装并配置Miniconda

miniconda官方文档&#xff1a; Miniconda — Anaconda 文档 官方文档中有讲到怎么安装Miniconda&#xff0c;如下&#xff1a; 以下是我得出的经验&#xff1a; 1. 新建新目录并下载和安装miniconda&#xff08;安装过程中&#xff0c;当提示是否继续时&#xff0c;一直按回…

蓝桥杯22年javaB组省赛真题

22年java_b组题目解析 写该博客既是为了分享题目解法&#xff0c;也是对之前写的题复习&#xff0c;毕竟已经24年了&#xff0c;写22年的题解 233&#x1f92d; 文章目录 22年java_b组题目解析A.星期计算(填空题)B.山&#xff08;填空题&#xff09;C.字符统计&#xff08;编…

day75 js 正则表达式 window对象轮播图片调用定时器

一 正则表达式: RegExp 对象: 对字符串执行模式匹配的强大工具。 1 创建正则表达式对象 let reg /模式/修饰符 修饰符 attributes 是一个可选的字符串&#xff0c;包含属性 "g"、"i" 和 "m"&#xff0c; …

Azure的VFP和虚拟IP地址

Azure 的Virtual filtering platform (VFP) 是Azure 网络地址转换,端口转换和端口分配的基础。 下面我们来深入介绍一下VFP的工作方式。 VFP的出站动作。 对于客户端地址作为虚拟IP的出站目的地址的时候,VFP 驱动会负责做以下两个动作。 源地址转换。端口地址转换。VFP 和 S…

20240325-1-HMM

HMM 直观理解 马尔可夫链&#xff08;英语&#xff1a;Markov chain&#xff09;&#xff0c;又称离散时间马尔可夫链&#xff08;discrete-time Markov chain&#xff0c;缩写为DTMC&#xff09;&#xff0c;因俄国数学家安德烈马尔可夫&#xff08;俄语&#xff1a;Андре…