将PCM格式存储成WAV格式文件
将PCM格式存储成WAV格式文件
WAV比PCM多44个字节(在文件头位置多)
摘自:https://blog.csdn.net/u012173922/article/details/78849076
前言:无论是文字,图像还是声音,都必须以一种特定的格式组织和存储起来,这样才能让显示器或播放器知道以怎样的一种方式去解析这些数据。
把PCM格式的数据存储成WAV格式数据的思路:先写头部,再写数据块。
WAV格式可以分成两个部分:
1.文件头,存储一些重要的参数信息,比如采样率,声道数,量化精度等等。
2.数据块,原始的PCM数据。
想要了解WAV格式的可以点击这里 点击打开链接
下面是WAV文件结构图
我们需要简单来说明一下这张图的结构:
可以分成三个部分:
第一部分RIFF : ChunkID 存储了“RIFF”字段,表示这是一个“RIFF”格式的文件。
ChunkSize 记录整个wav文件的字节数。
Format 存储了“WAVE”字段,表示这是一个wav文件。
第二部分fmt: 这部分的内容主要是记录一些关键参数,比如采样率,声道数,量化精度等等。
Subchunk1 ID 存储了“fmt”字段
Subchunk1 Size 存储“fmt”字段的长度
AudioFormat 存储 量化精度
Num Channels 存储声道数
SampleRate 存储采样率
ByteRate 存储比特率 SampleRate * NumChannels * BitsPerSample/8
BlockAlign == NumChannels * BitsPerSample/8
BitsPerSample 8 bits = 8, 16 bits = 16, etc.
第三部分data : 主要描述数据块
Subchunk2 ID 存储“data”字段
Subchunk2Size 记录存储的二进制原始音频数据的长度
data 存储二进制原始音频数据
我在网上找了一段wav写入头部的代码,亲测成功
-
byte[] header = new byte[44];
-
//RIFF WAVE Chunk
-
// RIFF标记占据四个字节
-
header[0] = 'R';
-
header[1] = 'I';
-
header[2] = 'F';
-
header[3] = 'F';
-
//数据大小表示,由于原始数据为long型,通过四次计算得到长度
-
header[4] = (byte) (totalDataLen & 0xff);
-
header[5] = (byte) ((totalDataLen >> 8) & 0xff);
-
header[6] = (byte) ((totalDataLen >> 16) & 0xff);
-
header[7] = (byte) ((totalDataLen >> 24) & 0xff);
-
//WAVE标记占据四个字节
-
header[8] = 'W';
-
header[9] = 'A';
-
header[10] = 'V';
-
header[11] = 'E';
-
//FMT Chunk
-
header[12] = 'f';
-
// 'fmt '标记符占据四个字节
-
header[13] = 'm';
-
header[14] = 't';
-
header[15] = ' ';//过渡字节
-
//数据大小
-
header[16] = 16; // 4 bytes: size of 'fmt ' chunk
-
header[17] = 0;
-
header[18] = 0;
-
header[19] = 0;
-
//编码方式 10H为PCM编码格式
-
header[20] = 1; // format = 1
-
header[21] = 0;
-
//通道数
-
header[22] = (byte) channels;
-
header[23] = 0;
-
//采样率,每个通道的播放速度
-
header[24] = (byte) (longSampleRate & 0xff);
-
header[25] = (byte) ((longSampleRate >> 8) & 0xff);
-
header[26] = (byte) ((longSampleRate >> 16) & 0xff);
-
header[27] = (byte) ((longSampleRate >> 24) & 0xff);
-
//音频数据传送速率,采样率*通道数*采样深度/8
-
header[28] = (byte) (byteRate & 0xff);
-
header[29] = (byte) ((byteRate >> 8) & 0xff);
-
header[30] = (byte) ((byteRate >> 16) & 0xff);
-
header[31] = (byte) ((byteRate >> 24) & 0xff);
-
// 确定系统一次要处理多少个这样字节的数据,确定缓冲区,通道数*采样位数
-
header[32] = (byte) (1 * 16 / 8);
-
header[33] = 0;
-
//每个样本的数据位数
-
header[34] = 16;
-
header[35] = 0;
-
//Data chunk
-
header[36] = 'd';//data标记符
-
header[37] = 'a';
-
header[38] = 't';
-
header[39] = 'a';
-
//数据长度
-
header[40] = (byte) (totalAudioLen & 0xff);
-
header[41] = (byte) ((totalAudioLen >> 8) & 0xff);
-
header[42] = (byte) ((totalAudioLen >> 16) & 0xff);
-
header[43] = (byte) ((totalAudioLen >> 24) & 0xff);
-
out.write(header, 0, 44);
解决了最困难的点,下面的工作就好实现了。
下面是完成的代码:
-
private void writeWav() {
-
fileTarget = new File(file, "audiotest.pcm");
-
fileWav = new File(file, "audiotest.wav");
-
-
if (!fileTarget.exists()) {
-
Log.e("tag", "目标文件不存在");
-
return;
-
}
-
DataInputStream dataInputStream = null;
-
DataOutputStream dataOutputStream = null;
-
try {
-
dataInputStream = new DataInputStream(new BufferedInputStream(new FileInputStream(fileTarget)));
-
dataOutputStream = new DataOutputStream(new BufferedOutputStream(new FileOutputStream(fileWav)));
-
int len = dataInputStream.available();
-
long totalAudioLen = 0;
-
long totalDataLen = totalAudioLen + 36;
-
long longSampleRate = 44100;
-
int channels = 1;
-
long byteRate = 16 * longSampleRate * channels / 8;
-
//写wav头部
-
writeWavHeader(dataOutputStream, totalAudioLen, totalDataLen, longSampleRate, channels, byteRate);
-
byte[] bytes = new byte[bufferSize];
-
int lenthg = -1;
-
while ((lenthg = dataInputStream.read(bytes)) != -1) {
-
dataOutputStream.write(bytes, 0, lenthg);
-
}
-
} catch (FileNotFoundException e) {
-
e.printStackTrace();
-
} catch (IOException e) {
-
e.printStackTrace();
-
} finally {
-
try {
-
dataInputStream.close();
-
dataOutputStream.close();
-
} catch (IOException e) {
-
e.printStackTrace();
-
}
-
}
-
}
-
private byte[] writeWavHeader(DataOutputStream dataOutputStream, long totalAudioLen, long totalDataLen, long longSampleRate,
-
int channels, long byteRate) throws IOException {
-
byte[] header = new byte[44];
-
//RIFF WAVE Chunk
-
// RIFF标记占据四个字节
-
header[0] = 'R';
-
header[1] = 'I';
-
header[2] = 'F';
-
header[3] = 'F';
-
//数据大小表示,由于原始数据为long型,通过四次计算得到长度
-
header[4] = (byte) (totalDataLen & 0xff);
-
header[5] = (byte) ((totalDataLen >> 8) & 0xff);
-
header[6] = (byte) ((totalDataLen >> 16) & 0xff);
-
header[7] = (byte) ((totalDataLen >> 24) & 0xff);
-
//WAVE标记占据四个字节
-
header[8] = 'W';
-
header[9] = 'A';
-
header[10] = 'V';
-
header[11] = 'E';
-
//FMT Chunk
-
header[12] = 'f';
-
// 'fmt '标记符占据四个字节
-
header[13] = 'm';
-
header[14] = 't';
-
header[15] = ' ';//过渡字节
-
//数据大小
-
header[16] = 16; // 4 bytes: size of 'fmt ' chunk
-
header[17] = 0;
-
header[18] = 0;
-
header[19] = 0;
-
//编码方式 10H为PCM编码格式
-
header[20] = 1; // format = 1
-
header[21] = 0;
-
//通道数
-
header[22] = (byte) channels;
-
header[23] = 0;
-
//采样率,每个通道的播放速度
-
header[24] = (byte) (longSampleRate & 0xff);
-
header[25] = (byte) ((longSampleRate >> 8) & 0xff);
-
header[26] = (byte) ((longSampleRate >> 16) & 0xff);
-
header[27] = (byte) ((longSampleRate >> 24) & 0xff);
-
//音频数据传送速率,采样率*通道数*采样深度/8
-
header[28] = (byte) (byteRate & 0xff);
-
header[29] = (byte) ((byteRate >> 8) & 0xff);
-
header[30] = (byte) ((byteRate >> 16) & 0xff);
-
header[31] = (byte) ((byteRate >> 24) & 0xff);
-
// 确定系统一次要处理多少个这样字节的数据,确定缓冲区,通道数*采样位数
-
header[32] = (byte) (1 * 16 / 8);
-
header[33] = 0;
-
//每个样本的数据位数
-
header[34] = 16;
-
header[35] = 0;
-
//Data chunk
-
header[36] = 'd';//data标记符
-
header[37] = 'a';
-
header[38] = 't';
-
header[39] = 'a';
-
//数据长度
-
header[40] = (byte) (totalAudioLen & 0xff);
-
header[41] = (byte) ((totalAudioLen >> 8) & 0xff);
-
header[42] = (byte) ((totalAudioLen >> 16) & 0xff);
-
header[43] = (byte) ((totalAudioLen >> 24) & 0xff);
-
dataOutputStream.write(header, 0, 44);
-
return header;
-
}
站在巨人的肩膀上
请多多指点
接下来一篇文章我会将带来 如何解析WAV格式文件
将PCM格式存储成WAV格式文件的更多相关文章
- c# Use NAudio Library to Convert MP3 audio into WAV audio(将Mp3格式转换成Wav格式)
Have you been in need of converting mp3 audios to wav audios? If so, the skill in this article prov ...
- iOS: lame框架将PCM录音转成MP3格式
lame框架将PCM录音转成MP3格式 1.lame下载地址:https://github.com/rbrito/lame,它是一个不可执行的文件,需要借助build-lame.sh脚本将其编译成.a ...
- Swift iOS实现把PCM语音转成MP3格式
最近折腾了swift的语音录制识别和转码,这块还是比较坑的,由于语音识别的准确度实测大概也就80%左右,所以还是需要上传录音文件啊.首先是用讯飞语音SDK实现语音录制和识别(语音听写),第一个坑是讯飞 ...
- 分别用Excel和python进行日期格式转换成时间戳格式
最近在处理一份驾驶行为方面的数据,其中要用到时间戳,因此就在此与大家一同分享学习一下. 1.什么是时间戳? 时间戳是指格林威治时间1970年01月01日00时00分00秒(北京时间1970年01月01 ...
- 怎样将M4A音频格式转换成MP3格式
因为MP3音频格式应用的广泛性,所以很多时候我们都需要将不同的音频格式转换成MP3格式的,那么如果我们需要将M4A音频格式转换成MP3格式,我们应该怎样进行实现呢?下面我们就一起来看一下吧. 操作步骤 ...
- 09: xmltodict 模块将xml格式转成json格式
1.1 : xmltodict 模块将xml格式转成json格式 <?xml version="1.0"?> <!--#版本号--> <data> ...
- python脚本实现音频m4a格式转成MP3格式
群里看到有人询问:谁会用python将微信音频文件后缀m4a格式转成mp3格式,毫不犹豫回了句:我会.然后就私下聊起来了 解决方法介绍如下: 工具:windows系统,python2.7,转换库ffm ...
- linux环境下deb格式 转换成rpm格式
linux环境下deb格式 转换成rpm格式 使用alien工具转换deb格式到rpm格式 alien_8.87.tar.gz 下载alien_8.87.tar.gz [root@mysqlnode2 ...
- Excel中将时间格式转化成时间戳格式
时间戳转成正常日期的公式:C1=(A1+8*3600)/86400+70*365+19其中A1表示当时的1249488000时间戳数值其中C1就是所需的日期格式,C1单元格属性改成日期格式就可以了.正 ...
随机推荐
- List list = new ArrayList();和ArrayList list=new ArrayList();的区别
List是一个接口,而ArrayList 是一个类. ArrayList 继承并实现了List.List list = new ArrayList();这句创建了一个ArrayList的对象后把上溯到 ...
- IO多路复用的作用?
I/O多路复用实际上就是用select, poll, epoll监听多个io对象,当io对象有变化(有数据)的时候就通知用户进程.好处就是单个进程可以处理多个socket.当然具体区别我们后面再讨论, ...
- Error Permission denied when running brew cleanup
Error Permission denied when running brew cleanup When I try to run `brew cleanup` I get: Warning: S ...
- 从0到N建立高性价比的大数据平台(转载)
2016-07-29 14:13:23 钱曙光 阅读数 794 原文链接:https://blog.csdn.net/qiansg123/article/details/80124521 声明:本文为 ...
- Linux 命令 ipcs/ipcrm
ipcs 1. 命令格式 ipcs [resource-option] [output-format] ipcs [resource-option] -i id 2. 命令功能 提供IPC设备的信息 ...
- 自用ajax的json请求
function singlePriceSubmit(){ var json={}; json["area"]=areaStr1; json["goodCateIdStr ...
- c函数指针和指针函数如何使用何定义;如何调用使用
#include <stdio.h> int * sum(int x); //声明一个 指针函数 返回类型位一个指针变量 可以通过*p来获取值 int (*pfun)(int,int);/ ...
- c语言数组的概念和指针的加减使用
//数组变量名:就是一个地址:就是数组首元素的地址#include <stdio.h> int main(void) { int age[5] = {10,50,100,22,44}; / ...
- 【转】Linux开机启动管理---systemd使用
常用命令 使某服务自动启动 systemctl enable httpd.service 使某服务不自动启动 systemctl disable httpd.service 检查服务状态 syste ...
- 安装vs code之后,win+e快捷键打开的是vs code,而不是文件管理器,解决方法
安装vs code之后,win+e快捷键打开的是vs code,而不是文件管理器,解决方法 xdg-mime default dde-file-manager.desktop inode/direct ...