使用Python子进程通信方法时如何获取退出代码?

Car*_*tem 172 python subprocess

使用Python的subprocess模块和communicate()方法时如何检索退出代码?

相关代码:

import subprocess as sp
data = sp.Popen(openRTSP + opts.split(), stdout=sp.PIPE).communicate()[0]
Run Code Online (Sandbox Code Playgroud)

我应该这样做吗?

Eli*_*sky 244

Popen.communicate将在returncode完成后设置属性(*).这是相关的文档部分:

Popen.returncode 
  The child return code, set by poll() and wait() (and indirectly by communicate()). 
  A None value indicates that the process hasn’t terminated yet.

  A negative value -N indicates that the child was terminated by signal N (Unix only).
Run Code Online (Sandbox Code Playgroud)

所以你可以这样做(我没有测试它,但它应该工作):

import subprocess as sp
child = sp.Popen(openRTSP + opts.split(), stdout=sp.PIPE)
streamdata = child.communicate()[0]
rc = child.returncode
Run Code Online (Sandbox Code Playgroud)

(*)这是因为它的实现方式:在设置线程以读取子流后,它只是调用wait.

  • 这个例子对我有所帮助,但是如果示例没有将"标准的导入子过程作为sp"模式导入标准为晦涩的缩写,那将会很好.虽然这会使其后面的代码缩小8个字符,但也会使其难以理解和重用. (23认同)
  • @uglycoyote没有规则说你必须复制和粘贴.只要你想要重新输入它,它就像4行一样. (11认同)
  • @uglycoyote您也可以将其编辑为类似于`from subprocess import Popen`,然后仅使用`Popen`代替`subprocess(or sp).Popen`,我想这可能会增加可读性并缩短行数 (4认同)
  • 是的...必须调用 `process.communicate()` 然后将 `returncode` 分配给某个变量。如果在调用 `communicate` 之前完成赋值,则为 `None`。 (2认同)

tri*_*eee 21

只是指出一个常见的误解,您应该Popen尽可能避免。引用文档,

调用子流程的推荐方法是将该run()函数用于它可以处理的所有用例。Popen对于更高级的用例,可以直接使用底层 接口。

如果您只想运行一个子进程并等待它完成,那么这是一行代码及其subprocess.run遗留兄弟subprocess.calland subprocess.check_output,并且您不需要复制/粘贴和/或了解所需的communicateandwait等方法的复杂性围绕低层Popen物体。

import subprocess

proc = subprocess.run(
    [openRTSP] + opts.split(),
    capture_output=True,
    # avoid having to explicitly encode
    text=True)
data = proc.stdout
result = proc.returncode
Run Code Online (Sandbox Code Playgroud)

如果您不想捕获该过程的输出,可以替换capture_output=Truestdout=subprocess.DEVNULL(对于 也可能类似stderr);如果两者都不存在,输出将简单地显示给用户,不受 Python 的控制。

另外,除非您的选项完全微不足道,否则通常将此处opts的常规字符串替换为了解如何处理带引号的字符串。split()shlex.split()


Mat*_*non 17

.poll() 将更新返回代码。

尝试

child = sp.Popen(openRTSP + opts.split(), stdout=sp.PIPE)
returnCode = child.poll()
Run Code Online (Sandbox Code Playgroud)

此外,在.poll()被调用之后,返回码在对象中可用child.returncode

  • 我认为您打算使用 .wait() 而不是 .poll(),根据文档:https://docs.python.org/3/library/subprocess.html。请注意, .wait() 需要一个可选的超时参数,这很方便。 (2认同)

Jos*_*ush 13

process.wait()打电话后使用process.communicate()
例如:

import subprocess

process = subprocess.Popen(['ipconfig', '/all'], stderr=subprocess.PIPE, stdout=subprocess.PIPE)
stdout, stderr = process.communicate()
exit_code = process.wait()
print(stdout, stderr, exit_code)
Run Code Online (Sandbox Code Playgroud)


Nou*_*him 8

您应首先确保该进程已完成运行,并且已使用该.wait方法读取了返回代码.这将返回代码.如果您想稍后访问它,它将存储.returncodePopen对象中.

  • `.communicate()`已经等待子进程终止. (24认同)

kha*_*hik 7

exitcode = data.wait().子进程将被阻止如果它写入标准输出/错误,和/或从标准输入读取,并且没有对等.