4

我有一个小的 python 脚本,基本上如下所示:

import os
import psutil

def processtree():
    pid = os.getpid()
    # have to go two levels up to skip calling shell and 
    # get to actual parent process
    parent = psutil.Process(pid).parent().parent()

    print 'Parent %s [PID = %d]' % (parent.name(), parent.pid)
    print '        |'
    for child in parent.children(recursive=True):
        if child.pid != pid:
            print '        - Child %s [PID = %d]' % (child.name(), child.pid)
        else:
            print '        - Child %s [PID = %d] (Self)' % (child.name(), child.pid)

if '__name__' == '__main__':
    processtree()

当我在 Windows 上运行此脚本时bash,没有其他任何运行,我看到以下内容:

Parent bash.exe [PID = 5984]
       |
       - Child bash.exe [PID = 5008]
       |
       - Child python.exe [PID = 3736] (Self)

这个信息是正确的。父 bash 进程是 PID 5984,python 进程是 3736。现在,我运行sleep 10000 &它以使其作为 PID 5984 的子进程运行。我检查ps -aef | grep 5984它并在那里;:

$ ps -aef | grep 5984 | grep -v grep | grep -v ps
myuser    5984       1 con    May 12 /bin/bash
myuser    5080    5984 con  11:17:12 /bin/sleep
myuser    3948    5984 con  11:36:47 /bin/bash

但是,当我再次运行我的脚本时,它仍然显示:

Parent bash.exe [PID = 5984]
       |
       - Child bash.exe [PID = 7560]
       |
       - Child python.exe [PID = 5168] (Self)

它不会显示sleep为父 bash 进程的子进程,即使ps它显示为存在。

请注意,自从创建了一个新的调用 shell 后,bash.exe 子项的 PID 发生了变化(不知道为什么会发生这种情况,但我认为这不相关)。python解释器的PID,因为我再次调用了脚本python processtree.py

不知道我做错了什么,我已经盯着这个看了一段时间。任何帮助表示赞赏...

4

1 回答 1

3

从评论中发布,以便其他人不会将其视为未回答的开放性问题

您需要改为使用parent = psutil.Process(pid).parent().

在 Unix 中,fork和的组合exec导致第二个bash进程被睡眠替换。

Windows 基于创建进程的衍生模型,这是继承自 DEC VMS 的传统。(Dave Cutler 管理了 VMS 和 NT 的设计。许多前 DEC 工程师在 1988 年跟随他来到 Microsoft。) NT 内核实际上可以实现forkexec,它为SUA 子系统所做的。

于 2015-07-03T21:45:24.610 回答