将32位浮点数转换为16位PCM范围

时间:2017-05-09 22:49:13

标签: python audio

我有一些由javascript HTML5网络音频API生成的数据。它生成Float32Array,一个32位浮点数组,介于-1和1之间。我使用websocket将数据流式传输到我的服务器。

我需要将32位浮点数转换为-32768和+32767(16位有符号整数)之间的16位PCM范围。然后,这允许将数据用作wav文件。

我遇到转换问题。我怀疑答案是使用struct模块,但我无法获得正确的格式。

1 个答案:

答案 0 :(得分:7)

这是一个示例Python 2.7程序,它读取包含原始32位浮点音频样本的文件,并创建一个WAV文件,其中包含转换后的16位有符号整数样本的样本:

import sys
import array
import struct
import wave

def convert(fin, fout, chunk_size = 1024 * 1024):
    chunk_size *= 4    # convert from samples to bytes

    waveout = wave.open(fout, "wb")
    waveout.setparams((1, 2, 44100, 0, "NONE", ""))

    while True:
        raw_floats = fin.read(chunk_size)
        if raw_floats == "":
            return
        floats = array.array('f', raw_floats)
        samples = [sample * 32767
                   for sample in floats]
        raw_ints = struct.pack("<%dh" % len(samples), *samples)
        waveout.writeframes(raw_ints)

convert(open(sys.argv[1], "rb"), open(sys.argv[2], "wb"))

代码使用array.array将32位浮点样本转换为Python浮点数,因为它应该比struct.unpack快一点。它也使用本机机器字节顺序,就像Float32Array一样。无法使用array.array创建16位整数样本,因为无论本地机器顺序如何,它们都需要使用小端字节顺序。范围转换由简单的Python代码处理。