实时读取子进程中的标准输出

人气:155 发布:2023-01-03 标签: python windows subprocess popen

问题描述

给定此代码片段:

from subprocess import Popen, PIPE, CalledProcessError


def execute(cmd):
    with Popen(cmd, shell=True, stdout=PIPE, bufsize=1, universal_newlines=True) as p:
        for line in p.stdout:
            print(line, end='')

    if p.returncode != 0:
        raise CalledProcessError(p.returncode, p.args)

base_cmd = [
    "cmd", "/c", "d:\virtual_envs\py362_32\Scripts\activate",
    "&&"
]
cmd1 = " ".join(base_cmd + ['python -c "import sys; print(sys.version)"'])
cmd2 = " ".join(base_cmd + ["python -m http.server"])

如果我运行execute(cmd1),则会毫无问题地打印输出。

但是,如果我运行execute(cmd2),则不会打印任何内容,这是为什么?我如何修复它以便可以实时查看HTTP.SERVER的输出。

另外,for line in p.stdout是如何在内部评估的?它是不是某种无休止的循环,直到到达标准的Eof或其他什么?

这个话题在本文中已经讨论过几次了,但我还没有找到Windows解决方案。上述代码片段来自answer,我正在从Virtualenv(在Win7上为python3.6.2-32bit)运行HTTP.SERVER

推荐答案

如果要从正在运行的子进程中连续读取,则必须使该进程的输出无缓冲。您的子进程是一个Python程序,这可以通过将-u传递给解释器来完成:

python -u -m http.server

这是它在Windows框上的外观。

19