如何确定通过 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的主要内容,如果未能解决你的问题,请参考以下文章