4

我在应用程序中使用SoX 。该应用程序使用它对音频文件应用各种操作,例如修剪。

这工作正常:

from subprocess import Popen, PIPE

kwargs = {'stdin': PIPE, 'stdout': PIPE, 'stderr': PIPE}

pipe = Popen(['sox','-t','mp3','-', 'test.mp3','trim','0','15'], **kwargs)
output, errors = pipe.communicate(input=open('test.mp3','rb').read())
if errors:
    raise RuntimeError(errors)

但是,这将导致大文件出现问题,因为read()将完整的文件加载到内存中;这很慢并且可能导致管道的缓冲区溢出。存在一种解决方法:

from subprocess import Popen, PIPE
import tempfile
import uuid
import shutil
import os

kwargs = {'stdin': PIPE, 'stdout': PIPE, 'stderr': PIPE}
tmp = os.path.join(tempfile.gettempdir(), uuid.uuid1().hex + '.mp3')

pipe = Popen(['sox','test.mp3', tmp,'trim','0','15'], **kwargs)
output, errors = pipe.communicate()

if errors:
    raise RuntimeError(errors)

shutil.copy2(tmp, 'test.mp3')
os.remove(tmp)

所以问题如下:除了为 Sox C API 编写 Python 扩展之外,还有其他方法可以替代这种方法吗?

4

1 回答 1

5

SoX 的 Python 包装器已经存在:sox。也许最简单的解决方案是切换到使用它,而不是通过subprocess.

以下使用包实现了您在示例中想要的内容sox(请参阅文档),并且应该在Python 2.7、3.43.5LinuxmacOS上工作(它也可能在 Windows 上工作,但我无法测试,因为我没有'无权访问 Windows 框):

>>> import sox
>>> transformer = sox.Transformer()  # create transformer 
>>> transformer.trim(0, 15)  # trim the audio between 0 and 15 seconds 
>>> transformer.build('test.mp3', 'out.mp3')  # create the output file 

注意:这个答案曾经提到不再维护的pysox包。感谢@erik 的提示。

于 2012-10-21T16:56:03.287 回答