python stream subprocess stdout和stderr zip不起作用

时间:2016-01-13 21:45:26

标签: python python-3.x subprocess

已经有很多答案可以解决如何做到这一点,但我的主要问题是为什么这种方法不起作用?

我正在尝试"直播"来自子进程的stdout和stderr。我可以这样做:

import sys
import subprocess

def run_command(cmd):
    process = subprocess.Popen(cmd, stderr=subprocess.PIPE, stdout=subprocess.PIPE)

    for out in iter(process.stdout.readline, b''):
        print(out)

    for err in iter(process.stderr.readline, b''):
        print(err)

run_command(['echo', 'hello', 'world']) # should print hello world
run_command(['rm', 'blarg223'])  # Should print to stderr (file doesnt exist)

这可以给我结果:

b'hello world\n'
b'rm: cannot remove \xe2\x80\x98blarg223\xe2\x80\x99: No such file or directory\n'

然而,这会导致一个问题,因为它实际上只在stdout中生成流,然后将所有错误打印为结束。所以我想我可以用以下方法解决这个问题:

def run_command(cmd):
    process = subprocess.Popen(cmd, stderr=subprocess.PIPE, stdout=subprocess.PIPE)

    for out, err in zip(iter(process.stdout.readline, b''), iter(process.stderr.readline, b'')):
        print(out)
        print(b'Error: ' + err)

但是,这不会产生任何输出。为什么使用zip不起作用?

1 个答案:

答案 0 :(得分:2)

当其中一个迭代器完成时,

zip停止。

在您给出的每个示例中,一个流(stdout / stderr)为空。所以拉链不会产生任何效果。

要解决此问题,您应该使用itertools.zip_longest