【问题标题】:Modify volume while streaming with PyAudio使用 PyAudio 在流式传输时修改音量
【发布时间】:2016-08-08 10:17:05
【问题描述】:

我正在尝试流式传输 .WAV 文件,同时剥离其中一个声道,因此它只在一个扬声器上播放,同时降低音量。

我使用 PyAudio 的原因是我必须能够

  • 一次流式传输多个文件
  • 通过特定扬声器播放声音(左/右)
  • 中断声音文件的播放

并且无法找到任何其他允许这些功能的 python 库。

我有以下代码,从 PyAudio 的流媒体示例修改而来。它从一个扬声器播放 WAV 文件,但我无法找到以任何方式修改音量的方法。

"""PyAudio Example: Play a wave file."""

import pyaudio
import wave
import sys 

CHUNK = 1024

if len(sys.argv) < 2:
    print("Plays a wave file.\n\nUsage: %s filename.wav" % sys.argv[0])
    sys.exit(-1)

wf = wave.open(sys.argv[1], 'rb')

# instantiate PyAudio (1)
p = pyaudio.PyAudio()

# open stream (2)
sampwidth = wf.getsampwidth()
stream = p.open(format=p.get_format_from_width(sampwidth),
                channels=wf.getnchannels(),
                rate=wf.getframerate(),
                output=True)

# read data
data = bytearray(wf.readframes(CHUNK))

# play stream (3)
while len(data) > 0:
    #remove left channel
    for i in range(0, len(data)):
        if i % (sampwidth*2) < sampwidth:
            data[i] = 0

    #reduce volume by 50%
    for i in range(0, len(data)):
        # use magic???
        pass

    stream.write(bytes(data))
    data = wf.readframes(CHUNK)
    if len(data) > 0:
        data = bytearray(data)

# stop stream (4)
stream.stop_stream()
stream.close()

# close PyAudio (5)
p.terminate()

如果有人可以给我任何指示,我将不胜感激。我觉得答案需要比我目前拥有的更好地了解 WAV 文件的格式。我尝试了data[i] = int(data[i]/2) 和其他类似的简单算法,但输出只是失真,这表明 WAV 文件格式并不像我想象的那么简单。

【问题讨论】:

  • 您的简单算术不起作用,因为 databytearray 并且没有定义这些操作(它们实际上没有意义,因为它只是字节,而不是数字)。你愿意使用 NumPy 吗?使用 NumPy 数组将使这些操作成为可能。

标签: python wav pyaudio


【解决方案1】:

您可以使用subprocess 模块:

来自subprocess 导入调用

 call(["amixer", "-D", "pulse", "sset", "Master", "0%"])

将音量提高 10%:

call(["amixer", "-D", "pulse", "sset", "Master", "10%+"])

将音量降低 10%:

call(["amixer", "-D", "pulse", "sset", "Master", "10%-"])

【讨论】:

    【解决方案2】:

    您可以为此使用numpy,而无需调用子进程。代码示例:

    import numpy
    
    def audio_datalist_set_volume(datalist, volume):
        """ Change value of list of audio chunks """
        sound_level = (volume / 100.)
    
        for i in range(len(datalist)):
            chunk = numpy.fromstring(datalist[i], numpy.int16)
    
            chunk = chunk * sound_level
    
            datalist[i] = chunk.astype(numpy.int16)
    

    注意:datalist 是您的声音阵列

    要使用它,只需将声音数据传递给函数并设置音量。

    【讨论】:

    • 当我问这个问题时,我工作的公司 3 年前已经停止了开发,但我仍然感谢这个建议
    【解决方案3】:

    您可能想结帐pydub。它使用 pyaudio 并且有一个简单的方法来改变音量。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2019-06-27
      • 2019-08-01
      • 1970-01-01
      • 1970-01-01
      • 2010-10-23
      • 1970-01-01
      • 1970-01-01
      • 2012-04-03
      相关资源
      最近更新 更多