2

我遇到了一个问题,即应该几乎立即返回的子进程调用最终阻塞(实际的子进程已失效且未清理)。这似乎只发生在我也使用线程时。下面的程序很好地再现了我所看到的。通过运行它,您会看到主机名被打印了很多次,数字在整个范围内循环,直到它被阻塞。我怀疑一旦线程完成,不存在的进程将被清理,然后它会再次发生。

任何人都了解正在发生的事情以及如何在不发生这种情况的情况下同时使用线程和子进程?我在 Ubuntu 11.4 上使用 Python 2.7.1+。

import os
import subprocess
import threading

class DummyThread(threading.Thread):

    def run(self):
        proc = subprocess.Popen(['sleep', '100'])
        proc.wait()

while True:
    dummy = DummyThread()
    dummy.start()
    for i in range(10):
        print i
        subprocess.check_call(['hostname'])

运行它会导致子进程失效并阻塞主线程:

user     28665 10543  1 22:55 pts/2        python blocker.py
user     28667 28665  0 22:55 pts/2        sleep 100
user     28679 28665  0 22:55 pts/2        sleep 100
user     28691 28665  0 22:55 pts/2        sleep 100
user     28703 28665  0 22:55 pts/2        sleep 100
user     28715 28665  0 22:55 pts/2        sleep 100
user     28727 28665  0 22:55 pts/2        sleep 100
user     28739 28665  0 22:55 pts/2        sleep 100
user     28751 28665  0 22:55 pts/2        sleep 100
user     28763 28665  0 22:55 pts/2        sleep 100
user     28775 28665  0 22:55 pts/2        sleep 100
user     28787 28665  0 22:55 pts/2        sleep 100
user     28799 28665  0 22:55 pts/2        sleep 100
user     28800 28665  0 22:55 pts/2        [hostname] <defunct>
4

1 回答 1

2

我在 ubuntu 12.04 (python 2.7.3) 和 OS X 10.8 (python 2.7.2) 上重现了这个问题。

如果在调用 subprocess.Popen 和 subprocess.check_call 时设置 close_fds=True,问题就会消失。

我不确定这意味着什么,但它可能会帮助有更多知识的人完全回答这个问题。

于 2012-10-29T22:55:48.267 回答