python并行进程返回退出代码

Posted

技术标签:

【中文标题】python并行进程返回退出代码【英文标题】:python parallel processes return exit code 【发布时间】:2017-03-01 00:16:52 【问题描述】:

让我们看看我能否说清楚...我是一个 Python 初学者,所以请耐心等待,这是我的第一个 Python 程序(尽管我熟悉其他几种语言的基本脚本)。我已经搜索了好几个小时,我确信这个问题的答案相当简单,但我还没有让它正常工作。

我正在编写一个应该启动多个命令行进程的代码,当每个进程完成时,我想更新 QTableWidget 中的一个单元格。该表有一行供每个进程运行,每一行都有一个单元格表示进程的“状态”。

如果我只做一个 for 循环,每行使用subprocess.call() 生成一个进程,我可以运行这个没有问题,但是这太线性了,我想同时将它们全部关闭而不是挂起程序每个循环周期。我一直在研究子流程文档,并且很难使用它。我知道我需要使用subprocess.Popen(这将防止我的程序在进程运行时挂起,因此我可以生成多个实例)。我遇到麻烦的地方是取回退出代码,以便我可以更新我的表,而无需挂起程序 - 例如使用 subprocess.wait() 后跟 subprocess.returncode 仍然只是坚持到过程完成。我需要一种“当过程完成时,检查退出代码并运行更新 QTableWidget 的函数。”

我确实发现这两个帖子似乎让我朝着正确的方向前进,但并没有完全让我到达那里:

Understanding Popen.communicate

How to get exit code when using Python subprocess communicate method?

希望这是有道理的。这是我的代码的简化版本,我意识到它是半生不熟和半坏的,但我已经搞砸了一个多小时,我已经忘记了一些事情......

import os, subprocess

ae_app = 'afterfx'
ae_path = os.path.join('C:/Program Files/Adobe/Adobe After Effects CC 2015/Support Files', ae_app + ".exe")
filename = "E:/Programming/Python/Archive tool/talk.jsx"
commandLine = 'afterfx -noui -r ' + filename

processList = [commandLine]
processes = []

for process in processList:
    f = os.tmpfile()
    aeProcess = subprocess.Popen(process, executable=ae_path, stdout=f)
    processes.append((aeProcess, f))

for aeProcess, f in processes:
    # this is where I need serious help...
    aeProcess.wait()
    print "the line is:"
    print aeProcess.returncode
斯宾塞

【问题讨论】:

【参考方案1】:

你提到了 PyQt,所以你可以使用 PyQt 的 QProcess 类。

def start_processes(self, process_list):
    for cmd, args in process_list:
        proc = QProcess(self)
        proc.finished.connect(self.process_finished)
        proc.start(cmd, args)

def process_finished(self, code, status):
    # Do something

更新:添加了完整的工作示例。适用于 PyQt4 和 PyQt5(仅切换注释第 3 行和取消注释第 4 行)

sleeper.py

import sys
from time import sleep
from datetime import datetime as dt

if __name__ == '__main__':
    x = int(sys.argv[1])
    started = dt.now().time()
    sleep(x)
    ended = dt.now().time()
    print('Slept for: , started: , ended: '.format(x, started, ended))
    sys.exit(0)

main.py

import sys

from PyQt5 import QtCore, QtWidgets
# from PyQt4 import QtCore, QtGui as QtWidgets


class App(QtWidgets.QMainWindow):
    cmd = r'python.exe C:\_work\test\sleeper.py '

    def __init__(self):
        super(App, self).__init__()
        self.setGeometry(200, 200, 500, 300)
        self.button = QtWidgets.QPushButton('Start processes', self)
        self.button.move(20, 20)
        self.editor = QtWidgets.QTextEdit(self)
        self.editor.setGeometry(20, 60, 460, 200)

        self.button.clicked.connect(self.start_proc)

    def start_proc(self):
        for x in range(5):
            proc = QtCore.QProcess(self)
            proc.finished.connect(self.finished)
            proc.start(self.cmd.format(x))

    def finished(self, code, status):
        self.editor.append(str(self.sender().readAllStandardOutput()))


if __name__ == '__main__':
    app = QtWidgets.QApplication(sys.argv)
    gui = App()
    gui.show()
    app.exec_()

【讨论】:

哇,不知道 PyQt 中存在这个!我遇到了一个错误,但我认为这是因为我错误地使用了 QProcess。在 codumentation 中看起来它需要 2 个参数:“cmd”和“args”,这就是我看到你设置 process_list 的方式。所以我想我的问题是——你能把它退后一步,告诉我你将如何设置“process_list”吗?它应该是一个元组还是什么?现在它只包含每个命令的字符串,该字符串是命令行文本 process_list = (('some.exe', '--some_params'), ('another.exe','-p 10 -r 20')) 感谢您到目前为止的帮助,快到了! args 列表实际上必须是一个 QStringArray,我没有问题。我现在遇到的是,其中一个 args 是一个包含空格的文件路径,并且由于某种原因,它没有正确传递给进程(它不响应它)。从文档中:包含空格的参数用引号括起来。 所以我认为这是我的问题所在,因为我在没有引号的情况下运行子进程。有趣的是,如果我在 cmd 中用引号输入它,它工作正常......不知道下一步该做什么! 哈,想出了一个解决方法!回顾文档时,我意识到您不需要在单独的数组中提供参数,您可以使用 program 参数将它们直接粉碎,这样可以防止 pyqt 弄乱您的字符串! 啊!好吧,我用 QProcess 运行了所有东西,但现在我的 process_finished 函数没有运行。根据这个link,您无法从QProcess.startDetached() 获得已完成的信号!所以我认为这是错误的方向......回到子流程?

以上是关于python并行进程返回退出代码的主要内容,如果未能解决你的问题,请参考以下文章

Python 子进程——如何忽略退出代码警告?

如果任何作业在 python 中失败,如何退出进程?

python多进程并行代码

Matlab 立即返回退出代码

Spring批量返回自定义进程退出代码

进程被强制终止时的进程退出代码