我现在一直在修补pyaudio一段时间,试图扭转一个简单的波形文件但没有成功。
在(我的)理论中,我只需要通过文件从头到尾迭代,每次回调pyaudio(1024帧)从文件中的相应索引中获取音频数据,反转生成的字符串并播放它。
这是我的代码(只有pyaudio回调和文件处理,其余部分不受示例代码的影响):
import pyaudio
import wave
import time
import sys
if len(sys.argv) < 2:
print("Plays a wave file.\n\nUsage: %s filename.wav" % sys.argv[0])
sys.exit(-1)
index = 40*1024
wf = wave.open(sys.argv[1], 'rb')
wf.setpos(index)
p = pyaudio.PyAudio()
def callback(in_data, frame_count, time_info, status):
global index
data = wf.readframes(frame_count)
data = data[::-1]
index-=1024
wf.setpos(index)
return (data, pyaudio.paContinue)
stream = p.open(format=p.get_format_from_width(wf.getsampwidth()),
channels=wf.getnchannels(),
rate=wf.getframerate(),
output=True,
stream_callback=callback)
stream.start_stream()
while stream.is_active():
time.sleep(0.1)
stream.stop_stream()
stream.close()
wf.close()
p.terminate()
我知道当它到达文件开头会崩溃,但它应该播放40×1024帧的反向音频......
答案 0 :(得分:5)
如果你想要反转的文件足够小以适应内存,你最好的选择是完全加载它并反转数据,然后流式传输:
import pyaudio
import wave
wavefile_name = 'wavefile.wav'
wf = wave.open(wavefile_name, 'rb')
p = pyaudio.PyAudio()
stream = p.open(format =
p.get_format_from_width(wf.getsampwidth()),
channels = wf.getnchannels(),
rate = wf.getframerate(),
output = True)
full_data = []
data = wf.readframes(1024)
while data:
full_data.append(data)
data = wf.readframes(1024)
data = ''.join(full_data)[::-1]
for i in range(0, len(data), 1024):
stream.write(data[i:i+1024])
但是,如果文件太大而无法放入内存,则需要某种方式向后读取文件并将其输入音响系统。这是一个完全不同的问题,因为它涉及一些低级别的I / O编程来处理向后读取。
编辑:看到完整代码后,我没有看到任何错误。代码在我的计算机中正常运行,但只在播放结束时失败。但是,你可以做两件事。首先,您可以使用以下命令转到wavefile的末尾以向后播放整个声音:
wf = wave.open(sys.argv[1], 'rb')
index = wf.getnframes() - 1024
wf.setpos(index)
其次,您必须修改回调,以便在搜索头超出文件开头时不会失败:
def callback(in_data, frame_count, time_info, status):
global index
data = wf.readframes(frame_count)
data = data[::-1]
index-=1024
if index < 0:
return (data, pyaudio.paAbort)
else:
wf.setpos(max(index,0))
return (data, pyaudio.paContinue)
除此之外,它还可以。