5

这个问题与我的其他问题之一的这个答案有关。我在这个答案中被告知可以os.kill(pid, 0)用来检查子进程是否已终止。如果它仍在运行,None则返回。如果它已终止,OSError则引发 an。到目前为止,这一直运行良好,但现在我处于一种os.kill(pid, 0)仍然返回的情况,None尽管子进程已经终止。进程 ID 为 82430,在 OSX 的活动监视器中,我再也找不到具有此 ID 的进程。但是,当我打开 Python shell 并输入os.kill(82430, 0)时,它仍然返回None. 我不明白。

准确地说,我通过 Ajax GET 请求在 Django 视图中定期检查子进程的状态,如上所示。

服务器端的 Django 视图

def monitor_process(request):
    response_data = {}

    try:
        os.kill(int(request.GET['process_id']), 0)
        response_data['process_status'] = 'running'
    except OSError:
        response_data['process_status'] = 'finished'

    return HttpResponse(json.dumps(response_data), mimetype='application/json')

客户端的 Ajax 调用

var monitorProcess = function(processId) {

    $.ajax({
        dataType: 'json',
        type: 'GET',
        url: '/monitor_process/',
        data: {'process_id': processId},
        success: function(response) {

            if (response.process_status == 'running') {
                // Only this branch is always executed

                // ...
            }

            else if (response.process_status == 'finished') {
                // This branch never gets executed

                // Stop checking for process status
                // The periodic check was started by 
                // window.setInterval in a different Ajax call
                clearInterval(monitorProcessId);

                // ...

            }
        }
    });
}; 

尽管该过程在某些时候停止并在活动监视器中消失,但os.kill()显然无法识别它。为什么会这样?非常感谢!

4

1 回答 1

4

这与 python 无关,更多的是关于进程本身:每个正在运行的进程可能会产生一些具有不同 PID 的线程。如果父母没有被正确杀死,他们会留在那里(“僵尸线程”)。它们与它们的父级在同一 ID 下分组在一起。这就是为什么,在 python 中,你可以使用

os.killpg(pgid, sig) 

杀死所有组。你可以检查下

/proc/<PID>/task/ 

(您的实际进程 ID 在哪里)用于生成的任何线程

于 2012-11-27T23:37:41.963 回答