C++ 将音频PCM数据封装成wav文件

2024-08-29 05:48
文章标签 c++ 音频 数据 封装 pcm wav

本文主要是介绍C++ 将音频PCM数据封装成wav文件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

使用声音设备采集的声音数据通常是PCM数据,直接写入文件是无法播放的,通常的做法是将其封装成wav格式,这样播放器就能够识别且播放了。本文将介绍如何将PCM封装成wav的方法。


一、如何实现?

首先需要构造wav头部,wav文件音频信息全部保存在头部,我们要做的就是在PCM数据的前面加入wav头,并且记录PCM的相关参数。

1.定义头结构

只定义PCM格式的wav文件头

//WAV头部结构-PCM格式
struct WavPCMFileHeader;

2.预留头部空间

创建文件时预留头部空间

FILE*f = fopen(fileName.c_str(), "wb+");
//预留头部位置
fseek(f, sizeof(WavPCMFileHeader), SEEK_SET);

3.写入PCM数据

写入数据,并记录数据总长度。

fwrite(data, 1, dataLength, f);
//录数据总长度
_totalDataLength += dataLength;

4.写入头部信息

关闭文件时,回到起始位置写入头部信息

//写入头部信息
fseek(f, 0, SEEK_SET);
WavPCMFileHeader h(_channels, _sampleRate, _bitsPerSample, _totalDataLength);
fwrite(&h, 1, sizeof(h), f);
fclose(f);

二、完整代码

WavWapper.h

#pragma once
#include<string>
namespace AC {class  WavWapper {public:WavWapper();~WavWapper();/// <summary>/// 创建wav文件/// </summary>/// <param name="fileName">文件名</param>/// <param name="channels">声道数</param>/// <param name="sampleRate">采样率,单位hz</param>/// <param name="bitsPerSample">位深</param>void CreateWavFile(const std::string &fileName, int channels, int  sampleRate, int  bitsPerSample);/// <summary>/// 写入PCM数据/// </summary>/// <param name="data">PCM数据</param>/// <param name="dataLength">数据长度</param>void WriteToFile(unsigned char* data, int dataLength);/// <summary>/// 关闭文件/// </summary>void CloseFile();private:void* _file=nullptr;uint32_t _totalDataLength=0;int _channels;int _sampleRate;int _bitsPerSample;};
}

WavWapper.cpp

#include"WavWapper.h"
#include<stdio.h>
namespace AC {//WAV头部结构-PCM格式struct WavPCMFileHeader{struct RIFF {const	char rift[4] = { 'R','I', 'F', 'F' };uint32_t fileLength;const	char wave[4] = { 'W','A', 'V', 'E' };}riff;struct Format{const	char fmt[4] = { 'f','m', 't', ' ' };uint32_t blockSize = 16;uint16_t formatTag;uint16_t channels;uint32_t samplesPerSec;uint32_t avgBytesPerSec;uint16_t blockAlign;uint16_t  bitsPerSample;}format;struct  Data{const	char data[4] = { 'd','a', 't', 'a' };uint32_t dataLength;}data;WavPCMFileHeader() {}WavPCMFileHeader(int nCh, int  nSampleRate, int  bitsPerSample, int dataSize) {riff.fileLength = 36 + dataSize;format.formatTag = 1;format.channels = nCh;format.samplesPerSec = nSampleRate;format.avgBytesPerSec = nSampleRate * nCh * bitsPerSample / 8;format.blockAlign = nCh * bitsPerSample / 8;format.bitsPerSample = bitsPerSample;data.dataLength = dataSize;}};WavWapper::WavWapper(){}WavWapper::~WavWapper(){CloseFile();}void WavWapper::CreateWavFile(const std::string& fileName, int channels, int sampleRate, int bitsPerSample){if (!_file){_channels = channels;_sampleRate = sampleRate;_bitsPerSample = bitsPerSample;_totalDataLength = 0;_file = fopen(fileName.c_str(), "wb+");//预留头部位置fseek(static_cast<FILE*>(_file), sizeof(WavPCMFileHeader), SEEK_SET);}}void WavWapper::WriteToFile(unsigned char* data, int dataLength){fwrite(data, 1, dataLength, static_cast<FILE*>(_file));_totalDataLength += dataLength;}void WavWapper::CloseFile(){if (_file){if (_totalDataLength > 0){//写入头部信息fseek(static_cast<FILE*>(_file), 0, SEEK_SET);WavPCMFileHeader h(_channels, _sampleRate, _bitsPerSample, _totalDataLength);fwrite(&h, 1, sizeof(h), static_cast<FILE*>(_file));}fclose(static_cast<FILE*>(_file));_file = nullptr;}}
}

三、使用示例

#include "WavWapper.h"
#include<Windows.h>
int main()
{	AC::WavWapper ww;//创建wav文件,确保pcm声音格式与参数一致ww.CreateWavFile("sound.wav",2, 44100, 16);while (flag){//获取PCM数据//略//获取PCM数据-end//写入PCM数据ww.WriteToFile(data, dataLength);}//关闭文件ww.CloseFile();	
}

总结

以上就是今天要讲的内容,PCM封装成wav还是相对较简单的,只要了解wav头结构,然后自定义其头结构,然后再进行一定的测试,就可以实现这样一个功能。

这篇关于C++ 将音频PCM数据封装成wav文件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

MySQL快速复制一张表的四种核心方法(包括表结构和数据)

《MySQL快速复制一张表的四种核心方法(包括表结构和数据)》本文详细介绍了四种复制MySQL表(结构+数据)的方法,并对每种方法进行了对比分析,适用于不同场景和数据量的复制需求,特别是针对超大表(1... 目录一、mysql 复制表(结构+数据)的 4 种核心方法(面试结构化回答)方法 1:CREATE

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

C++打印 vector的几种方法小结

《C++打印vector的几种方法小结》本文介绍了C++中遍历vector的几种方法,包括使用迭代器、auto关键字、typedef、计数器以及C++11引入的范围基础循环,具有一定的参考价值,感兴... 目录1. 使用迭代器2. 使用 auto (C++11) / typedef / type alias

C++ scoped_ptr 和 unique_ptr对比分析

《C++scoped_ptr和unique_ptr对比分析》本文介绍了C++中的`scoped_ptr`和`unique_ptr`,详细比较了它们的特性、使用场景以及现代C++推荐的使用`uni... 目录1. scoped_ptr基本特性主要特点2. unique_ptr基本用法3. 主要区别对比4. u

C++11中的包装器实战案例

《C++11中的包装器实战案例》本文给大家介绍C++11中的包装器实战案例,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录引言1.std::function1.1.什么是std::function1.2.核心用法1.2.1.包装普通函数1.2.

C++多线程开发环境配置方法

《C++多线程开发环境配置方法》文章详细介绍了如何在Windows上安装MinGW-w64和VSCode,并配置环境变量和编译任务,使用VSCode创建一个C++多线程测试项目,并通过配置tasks.... 目录下载安装 MinGW-w64下载安装VS code创建测试项目配置编译任务创建 tasks.js

MySQL中的DELETE删除数据及注意事项

《MySQL中的DELETE删除数据及注意事项》MySQL的DELETE语句是数据库操作中不可或缺的一部分,通过合理使用索引、批量删除、避免全表删除、使用TRUNCATE、使用ORDERBY和LIMI... 目录1. 基本语法单表删除2. 高级用法使用子查询删除删除多表3. 性能优化策略使用索引批量删除避免