将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单元格属性改成日期格式就可以了.正 ...
随机推荐
- Python语言程序设计(3)--实例2-python蟒蛇绘制-turtle库
1. 2. 3.了解turtle库 Turtle,也叫海龟渲染器,使用Turtle库画图也叫海龟作图.Turtle库是Python语言中一个很流行的绘制图像的函数库.海龟渲染器,和各种三维软件都有着良 ...
- C语言二级指针间接赋值
重要意义:间接赋值的意义,实现了模块的功能划分,实现了软件作品的分层,使得模块接口和信息系统得以实现. 所谓二级指针就是指向指针的指针,其声明形式如下 int *p=NULL int **p1=NUL ...
- Hive 调优
今天总结本人在使用Hive过程中的一些优化技巧,希望给大家带来帮助.Hive优化最体现程序员的技术能力,面试官在面试时最喜欢问的就是Hive的优化技巧. 技巧1.控制reducer数量 下面的内容是我 ...
- 使用echarts生成海友网企业全国分布地图
不分类别的效果 不同分类的分布效果图 从海友网获取各个企业名单保存进mysql cmfishhelper.py 从下列网址得到各个企业名片的网址保存进表cmfish cds = get_cds() h ...
- [转载]pythonnet
python与c#的交互模块pythonnethttp://www.cnblogs.com/tester-zhenghan/p/5406521.html [集成IronPython] 添加CLR对象到 ...
- JSON字符串 拼接与解析
常用方式: json字符串拼接(目前使用过两种方式): 1.运用StringBuilder拼接 StringBuilder json = new StringBuilder(); json.appen ...
- mac系统提示 interactive intelligence 的恼人问题
处理 interacti intelligence 提示问题记录 二手购买了一台电脑,从最初的小白到现在稍微熟悉mac的使用, 一直困扰我的便是一个提示, 上图 困扰多年, 记录一下解决和尝试过程吧. ...
- java连数据库和数据库连接池踩坑日记(一)-------oracle连接的一些问题
最近接触oracle有点多,同时也在配置数据库连接池,坑也就踩多了,记录下. 事情还没有结束,没时间记录问题,很多事情都忘了,过了国庆再写的话可能就真的全忘了吧……而且不单单是数据库问题,还有一些数据 ...
- note_4.10
单位根反演 \[ \frac{1}{k}\sum_{i=0}^{k-1}\omega_k^{in}=[k|n] \] 所以 \[ \begin{equation} \begin{split} \sum ...
- linux 命令scp
scp命令网络传输文件 上传文件 scp 文件名 usename@10.233.23.100:Data/ 上传文件夹到服务器 scp -r 文件夹(不带/)usename@10.233.23.100: ...