java pcm到wav
我有一个pcm文件,我想把它转换成wav文件 是否有适合的api或代码?此资源帮助我将PCM数据解析为WAVE。我已经基于它构建了一个库,它对我来说很好。我知道一个叫做“OperateWav”的库,我用它来开发转换器(linux c/c++)在我第一次实习期间的第一个项目中。我不确定这个项目本身是否存在以及它是否支持java。实际上,wav文件只是在pcm原始数据上添加了一个wav格式头…这是我的代码java pcm到wav,java,wav,pcm,Java,Wav,Pcm,我有一个pcm文件,我想把它转换成wav文件 是否有适合的api或代码?此资源帮助我将PCM数据解析为WAVE。我已经基于它构建了一个库,它对我来说很好。我知道一个叫做“OperateWav”的库,我用它来开发转换器(linux c/c++)在我第一次实习期间的第一个项目中。我不确定这个项目本身是否存在以及它是否支持java。实际上,wav文件只是在pcm原始数据上添加了一个wav格式头…这是我的代码 /** * Write PCM data as WAV file * @param os
/**
* Write PCM data as WAV file
* @param os Stream to save file to
* @param pcmdata 8 bit PCMData
* @param srate Sample rate - 8000, 16000, etc.
* @param channel Number of channels - Mono = 1, Stereo = 2, etc..
* @param format Number of bits per sample (16 here)
* @throws IOException
*/
public void PCMtoFile(OutputStream os, short[] pcmdata, int srate, int channel, int format) throws IOException {
byte[] header = new byte[44];
byte[] data = get16BitPcm(pcmdata);
long totalDataLen = data.length + 36;
long bitrate = srate * channel * format;
header[0] = 'R';
header[1] = 'I';
header[2] = 'F';
header[3] = 'F';
header[4] = (byte) (totalDataLen & 0xff);
header[5] = (byte) ((totalDataLen >> 8) & 0xff);
header[6] = (byte) ((totalDataLen >> 16) & 0xff);
header[7] = (byte) ((totalDataLen >> 24) & 0xff);
header[8] = 'W';
header[9] = 'A';
header[10] = 'V';
header[11] = 'E';
header[12] = 'f';
header[13] = 'm';
header[14] = 't';
header[15] = ' ';
header[16] = (byte) format;
header[17] = 0;
header[18] = 0;
header[19] = 0;
header[20] = 1;
header[21] = 0;
header[22] = (byte) channel;
header[23] = 0;
header[24] = (byte) (srate & 0xff);
header[25] = (byte) ((srate >> 8) & 0xff);
header[26] = (byte) ((srate >> 16) & 0xff);
header[27] = (byte) ((srate >> 24) & 0xff);
header[28] = (byte) ((bitrate / 8) & 0xff);
header[29] = (byte) (((bitrate / 8) >> 8) & 0xff);
header[30] = (byte) (((bitrate / 8) >> 16) & 0xff);
header[31] = (byte) (((bitrate / 8) >> 24) & 0xff);
header[32] = (byte) ((channel * format) / 8);
header[33] = 0;
header[34] = 16;
header[35] = 0;
header[36] = 'd';
header[37] = 'a';
header[38] = 't';
header[39] = 'a';
header[40] = (byte) (data.length & 0xff);
header[41] = (byte) ((data.length >> 8) & 0xff);
header[42] = (byte) ((data.length >> 16) & 0xff);
header[43] = (byte) ((data.length >> 24) & 0xff);
os.write(header, 0, 44);
os.write(data);
os.close();
}
编辑:2016-01-11
public byte[] get16BitPcm(short[] data) {
byte[] resultData = new byte[2 * data.length];
int iter = 0;
for (double sample : data) {
short maxSample = (short)((sample * Short.MAX_VALUE));
resultData[iter++] = (byte)(maxSample & 0x00ff);
resultData[iter++] = (byte)((maxSample & 0xff00) >>> 8);
}
return resultData;
}
这应该很简单,因为WAV=元数据+PCM(按该顺序)。这应该起作用:
private void rawToWave(final File rawFile, final File waveFile) throws IOException {
byte[] rawData = new byte[(int) rawFile.length()];
DataInputStream input = null;
try {
input = new DataInputStream(new FileInputStream(rawFile));
input.read(rawData);
} finally {
if (input != null) {
input.close();
}
}
DataOutputStream output = null;
try {
output = new DataOutputStream(new FileOutputStream(waveFile));
// WAVE header
// see http://ccrma.stanford.edu/courses/422/projects/WaveFormat/
writeString(output, "RIFF"); // chunk id
writeInt(output, 36 + rawData.length); // chunk size
writeString(output, "WAVE"); // format
writeString(output, "fmt "); // subchunk 1 id
writeInt(output, 16); // subchunk 1 size
writeShort(output, (short) 1); // audio format (1 = PCM)
writeShort(output, (short) 1); // number of channels
writeInt(output, 44100); // sample rate
writeInt(output, RECORDER_SAMPLERATE * 2); // byte rate
writeShort(output, (short) 2); // block align
writeShort(output, (short) 16); // bits per sample
writeString(output, "data"); // subchunk 2 id
writeInt(output, rawData.length); // subchunk 2 size
// Audio data (conversion big endian -> little endian)
short[] shorts = new short[rawData.length / 2];
ByteBuffer.wrap(rawData).order(ByteOrder.LITTLE_ENDIAN).asShortBuffer().get(shorts);
ByteBuffer bytes = ByteBuffer.allocate(shorts.length * 2);
for (short s : shorts) {
bytes.putShort(s);
}
output.write(fullyReadFileToBytes(rawFile));
} finally {
if (output != null) {
output.close();
}
}
}
byte[] fullyReadFileToBytes(File f) throws IOException {
int size = (int) f.length();
byte bytes[] = new byte[size];
byte tmpBuff[] = new byte[size];
FileInputStream fis= new FileInputStream(f);
try {
int read = fis.read(bytes, 0, size);
if (read < size) {
int remain = size - read;
while (remain > 0) {
read = fis.read(tmpBuff, 0, remain);
System.arraycopy(tmpBuff, 0, bytes, size - remain, read);
remain -= read;
}
}
} catch (IOException e){
throw e;
} finally {
fis.close();
}
return bytes;
}
private void writeInt(final DataOutputStream output, final int value) throws IOException {
output.write(value >> 0);
output.write(value >> 8);
output.write(value >> 16);
output.write(value >> 24);
}
private void writeShort(final DataOutputStream output, final short value) throws IOException {
output.write(value >> 0);
output.write(value >> 8);
}
private void writeString(final DataOutputStream output, final String value) throws IOException {
for (int i = 0; i < value.length(); i++) {
output.write(value.charAt(i));
}
}
这一切是如何运作的
如您所见,WAV头是WAV和PCM文件格式之间的唯一区别。假设您正在录制16位PCM单声道音频(根据您的代码,您正在录制)。rawToWave函数只是将标题整齐地添加到WAV文件中,以便音乐播放器知道打开文件时会发生什么,然后在标题之后,它只从最后一位开始写入PCM数据
酷提示
如果你想改变声音的音调,或者制作一个语音转换应用程序,你所要做的就是增加/减少
writeInt(output,44100);//代码中的采样率
。减小它将告诉玩家以不同的速率播放,从而改变输出音高。只是一点额外的“好消息”而已。:) 我还不能留下评论,但请注意,devflow的答案中的get16bit pcm
方法[奇怪地]缩放输入数据。如果已经有16位pcm数据要写入wav文件,则该方法应如下所示:
File f1 = new File("/sdcard/44100Sampling-16bit-mono-mic.pcm"); // The location of your PCM file
File f2 = new File("/sdcard/44100Sampling-16bit-mono-mic.wav"); // The location where you want your WAV file
try {
rawToWave(f1, f2);
} catch (IOException e) {
e.printStackTrace();
}
public byte[] get16BitPcm(short[] data) {
byte[] resultData = new byte[2 * data.length];
int iter = 0;
for (short sample : data) {
resultData[iter++] = (byte)(sample & 0x00ff);
resultData[iter++] = (byte)((sample & 0xff00) >>> 8);
}
return resultData;
}
get16BitPcm(short[])
是否只生成一个byte[]
参数大小的两倍?如果是的话,什么是持久性?如果没有,它会做什么?@Scruffy是的,很抱歉遗漏了,method.,刚刚更新了我的答案。根据,似乎头中的字节16对于PCM应该始终是“16”,而不管每个样本的位数是多少。