Python:从subprocess运行的子进程中实时获取输出

有些时候,我们需要将某些程序放到子进程中去运行,以达到整合系统的目的。在Python中,一个非常好的选择就是使用subprocess模块,本模块为开辟子进程去执行子程序提供了统一的接口,更加便于学习和使用。

 

同时,对于在子进程里的程序,我们希望能够实时获取其输出,以在主进程中打印相关信息,使我们能够了解当前子程序的执行进度。对此,subprocess模块也提供了相应的参数,能够将子程序的标准输出和标准错误输出返回给主程序。

 

下面,我们就通过一个例子来说明这个功能。首先,我们需要一个用于模拟标准输出和标准错误输出的“子程序”——subprogram.py:

 

import sys
import time


for i in range(5):
    sys.stdout.write('Processing {}\n'.format(i))
    time.sleep(1)

for i in range(5):
    sys.stderr.write('Error {}\n'.format(i))
    time.sleep(1)


可以看到这个程序非常简单,分别向标准输出和标准错误写入了5条信息,并且输出之间有1秒的间隔。下面是驱动这个“子程序”运行的“主程序”——main.py:

 

 

import shlex
import subprocess

if __name__ == '__main__':
    shell_cmd = 'python3 subprogram.py'
    cmd = shlex.split(shell_cmd)
    p = subprocess.Popen(cmd, shell=False, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
    while p.poll() is None:
        line = p.stdout.readline()
        line = line.strip()
        if line:
            print('Subprogram output: [{}]'.format(line))
    if p.returncode == 0:
        print('Subprogram success')
    else:
        print('Subprogram failed')


可以看到,我们通过指定stderr=subprocess.STDOUT,将子程序的标准错误输出重定向到了标准输出,以使我们可以直接从标准输出中同时获取标准输出和标准错误的信息。运行这个程序,我们会期待main.py会每秒输出一次信息到控制台,但是事实上,我们直到等了10秒之后才一次性看到所有的10条输出。

 

产生这种现象的原因也非常简单,就是标准输出和标准错误有一个缓存的概念,它不会立即将程序的标准输出内容返回,而是会做一定的缓存,直到缓存满或者程序结束强制清空缓存时才输出。了解到问题的原因,解决问题的方法也就一目了然了,我们只需要在子程序中,每次输出后去手动清空一下缓存即可,以下是修改过的subprogram.py:

 

import sys
import time


for i in range(5):
    sys.stdout.write('Processing {}\n'.format(i))
    sys.stdout.flush()
    time.sleep(1)

for i in range(5):
    sys.stderr.write('Error {}\n'.format(i))
    sys.stderr.flush()
    time.sleep(1)


经过上述的修改之后,再次运行main.py程序,我们会看到,每秒会输出一条信息,达到了我们在主程序中,去追踪子程序执行过程的目的。

 

 

PS:测试环境是Python3.6.1 Mac版。

  • 17
    点赞
  • 56
    收藏
    觉得还不错? 一键收藏
  • 11
    评论
当使用Pythonsubprocess模块执行一个外部命令时,如果该命令返回了非零的退出状态码,会触发subprocess.CalledProcessError异常。这个异常会包含有关子进程返回状态码和错误输出的信息。 一般情况下,这个错误表示外部命令在执行时发生了错误。有几种可能的原因导致此错误: 1. 命令不存在或无法找到:如果子进程要执行的命令不存在,那么系统将返回一个非零的状态码。要解决此问题,请确保指定的命令被正确安装或命令的路径是正确的。 2. 输入或参数错误:有时候外部命令可能会对输入数据或参数进行验证,如果给定的数据或参数不符合要求,子进程可能会返回一个非零的状态码。在这种情况下,需要确认提供的数据和参数是正确的,并且符合外部命令的预期。 3. 权限问题:某些命令可能需要管理员权限才能执行,如果当前用户没有足够的权限运行这个命令,子进程将返回一个非零的状态码。要解决此问题,请确保使用了具有足够权限的用户运行Python程序。 为了处理这个异常,可以使用try-except语句来捕获CalledProcessError异常,并在异常处理块处理相关的错误信息。通过访问异常对象的output属性可以获取外部命令的错误输出,通过访问returncode属性可以得到子进程的返回状态码。 以下是一个处理subprocess.CalledProcessError异常的示例: ```python import subprocess try: # 执行外部命令,可能会触发异常 output = subprocess.check_output(["non_existing_command"]) except subprocess.CalledProcessError as e: # 处理异常 print("外部命令返回非零状态码:", e.returncode) print("错误输出信息:", e.output) ``` 通过适当地处理subprocess.CalledProcessError异常,我们可以增强程序的稳定性,并确保在发生子进程错误时我们能正确处理异常。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值