如何确定通过 os.system 启动的进程的 pid

Posted

技术标签:

【中文标题】如何确定通过 os.system 启动的进程的 pid【英文标题】:How to determine pid of process started via os.system 【发布时间】:2013-12-11 16:25:15 【问题描述】:

我想用一个程序启动几个子进程,即一个模块foo.py启动几个bar.py的实例。

由于我有时必须手动终止进程,因此我需要进程 ID 来执行 kill 命令。

即使整个设置非常“脏”,如果进程是通过os.system 启动的,是否有一种很好的pythonic 方法来获取进程'pid

foo.py:

import os
import time
os.system("python bar.py \"0\ &".format(str(argument)))
time.sleep(3)
pid = ???
os.system("kill -9 0".format(pid))

bar.py:

import time
print("bla")
time.sleep(10) % within this time, the process should be killed
print("blubb")

【问题讨论】:

"因此子进程Popen(["python", "bar.py", "arguments"]) 太慢了。" -- 你实际测量过吗? The question that you've linked 显示不同的情况,无论如何时间差是几毫秒。 是的,我已经测量过了。大约 20 到 30 毫秒,在我的具体情况下太慢了,因为我有一个滚动图形,必须每秒更新多次。 os.spawnl 给了我想要的功能,尽管它增加了大约一秒钟来启动每个进程,但没有停止我的主进程。它仍然不是理想的解决方案,但对我来说比 subprocess.Popen 更好。 你应该在问题中提到它(理想情况下,有一个完整的最小代码示例,其他人可以尝试)。当心spawn 继承所有可继承的文件描述符(Popen 关闭它们(Python 3))。您可以先启动进程并暂停它们,然后再取消暂停它们(发送输入或信号)吗?为什么不能在同一进程中使用bar.py 脚本?顺便说一句,与Popen 相比,为什么spawn 会在进程开始时增加一整秒? 关于我的程序的一些细节:一个连续(每秒几次)更新的图表从右到左滚动。在bar.py 中,进行了几次耗时的计算。虽然计算没有完成,但在图中写着“计算”。如果用户进行任何输入,则必须终止计算并开始新的计算。计算结果的快速得出并不重要,而图表确实可以流畅地移动。额外一秒的原因我还不清楚。 在进行了其他几项更改后,我意识到,我无法重现通过 Popen 启动进程的较长时间,也无法重现通过 Popen 启动进程时的额外秒数。我将更改问题以使接受的答案适合并撤销我的解决方案。很抱歉打扰大家——我不知道我之前的结果是从哪里来的。 【参考方案1】:

os.system 返回退出代码。它不提供子进程的pid。

使用subprocess 模块。

import subprocess
import time
argument = '...'
proc = subprocess.Popen(['python', 'bar.py', argument], shell=True)
time.sleep(3) # <-- There's no time.wait, but time.sleep.
pid = proc.pid # <--- access `pid` attribute to get the pid of the child process.

要终止进程,您可以使用terminate 方法或kill。 (无需使用外部kill程序)

proc.terminate()

【讨论】:

@SebastianWerk,根据os.spawn* documentation,“请注意,子进程模块为生成新进程和检索其结果提供了更强大的工具;使用该模块比使用这些函数更可取。...” 我知道,因此我首先使用它,但是在我的计算机上启动一个进程大约需要 20 毫秒,这有点太长了): @SebastianWerk,将您的问题的 EDIT 部分发布为答案并接受它而不是试图关闭您的问题怎么样? 如果您希望主进程(上面的代码)在生成新进程并写入 pid 文件后停止运行,是否可行?【参考方案2】:

分享我的解决方案以防它可以帮助其他人:

我从这个页面获取信息在后台运行一个 fortran exe。我尝试使用 os.forkpty 来获取它的 pid,但它没有给出我的进程的 pid。我不能使用子进程,因为我不知道它如何让我在后台运行我的进程。

在同事的帮助下,我发现了这个:

exec_cmd = 'nohup ./FPEXE & echo $! > /tmp/pid'

os.system(exec_cmd)

如果想将 pid 附加到同一个文件,请使用双箭头。

【讨论】:

【参考方案3】:

您可以改用os.forkpty(),它作为结果代码为您提供伪终端的pid 和fd。更多文档在这里:http://docs.python.org/2/library/os.html#os.forkpty

【讨论】:

谢谢,如果os.spawnl(...) 失败,我将使用它作为后备

以上是关于如何确定通过 os.system 启动的进程的 pid的主要内容,如果未能解决你的问题,请参考以下文章

python cmd命令调用

python调用OS.system结束进程问题?

子进程相对于 os.system 的优势

通过 os.system 推送

使用subprocessm模块管理进程

python 中os.system() 调用命令超时后,如何停止