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

相关文章

Spring 请求之传递 JSON 数据的操作方法

《Spring请求之传递JSON数据的操作方法》JSON就是一种数据格式,有自己的格式和语法,使用文本表示一个对象或数组的信息,因此JSON本质是字符串,主要负责在不同的语言中数据传递和交换,这... 目录jsON 概念JSON 语法JSON 的语法JSON 的两种结构JSON 字符串和 Java 对象互转

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

SpringBoot使用GZIP压缩反回数据问题

《SpringBoot使用GZIP压缩反回数据问题》:本文主要介绍SpringBoot使用GZIP压缩反回数据问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot使用GZIP压缩反回数据1、初识gzip2、gzip是什么,可以干什么?3、Spr

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

SpringBoot集成Milvus实现数据增删改查功能

《SpringBoot集成Milvus实现数据增删改查功能》milvus支持的语言比较多,支持python,Java,Go,node等开发语言,本文主要介绍如何使用Java语言,采用springboo... 目录1、Milvus基本概念2、添加maven依赖3、配置yml文件4、创建MilvusClient

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

SpringValidation数据校验之约束注解与分组校验方式

《SpringValidation数据校验之约束注解与分组校验方式》本文将深入探讨SpringValidation的核心功能,帮助开发者掌握约束注解的使用技巧和分组校验的高级应用,从而构建更加健壮和可... 目录引言一、Spring Validation基础架构1.1 jsR-380标准与Spring整合1

MySQL 中查询 VARCHAR 类型 JSON 数据的问题记录

《MySQL中查询VARCHAR类型JSON数据的问题记录》在数据库设计中,有时我们会将JSON数据存储在VARCHAR或TEXT类型字段中,本文将详细介绍如何在MySQL中有效查询存储为V... 目录一、问题背景二、mysql jsON 函数2.1 常用 JSON 函数三、查询示例3.1 基本查询3.2

SpringBatch数据写入实现

《SpringBatch数据写入实现》SpringBatch通过ItemWriter接口及其丰富的实现,提供了强大的数据写入能力,本文主要介绍了SpringBatch数据写入实现,具有一定的参考价值,... 目录python引言一、ItemWriter核心概念二、数据库写入实现三、文件写入实现四、多目标写入