Python子进程并行

sas*_*hab 18 python subprocess

我希望能够并行运行多个进程,并且能够随时使用stdout.我该怎么办?我是否需要为每个subprocess.Popen()呼叫运行线程,什么?

jfs*_*jfs 18

你可以在一个线程中完成.

假设您有一个随机打印行的脚本:

#!/usr/bin/env python
#file: child.py
import os
import random
import sys
import time

for i in range(10):
    print("%2d %s %s" % (int(sys.argv[1]), os.getpid(), i))
    sys.stdout.flush()
    time.sleep(random.random())
Run Code Online (Sandbox Code Playgroud)

并且您希望在输出可用后立即收集输出,您可以select在POSIX系统上 使用@zigg建议:

#!/usr/bin/env python
from __future__ import print_function
from select     import select
from subprocess import Popen, PIPE

# start several subprocesses
processes = [Popen(['./child.py', str(i)], stdout=PIPE,
                   bufsize=1, close_fds=True,
                   universal_newlines=True)
             for i in range(5)]

# read output
timeout = 0.1 # seconds
while processes:
    # remove finished processes from the list (O(N**2))
    for p in processes[:]:
        if p.poll() is not None: # process ended
            print(p.stdout.read(), end='') # read the rest
            p.stdout.close()
            processes.remove(p)

    # wait until there is something to read
    rlist = select([p.stdout for p in processes], [],[], timeout)[0]

    # read a line from each process that has output ready
    for f in rlist:
        print(f.readline(), end='') #NOTE: it can block
Run Code Online (Sandbox Code Playgroud)

更便携的解决方案(应该在Windows,Linux,OSX工作)可以使用阅读器线程每个流程,看到无阻塞读取Python中的subprocess.PIPE.

这是os.pipe()基于Unix和Windows的解决方案:

#!/usr/bin/env python
from __future__ import print_function
import io
import os
import sys
from subprocess import Popen

ON_POSIX = 'posix' in sys.builtin_module_names

# create a pipe to get data
input_fd, output_fd = os.pipe()

# start several subprocesses
processes = [Popen([sys.executable, 'child.py', str(i)], stdout=output_fd,
                   close_fds=ON_POSIX) # close input_fd in children
             for i in range(5)]
os.close(output_fd) # close unused end of the pipe

# read output line by line as soon as it is available
with io.open(input_fd, 'r', buffering=1) as file:
    for line in file:
        print(line, end='')
#
for p in processes:
    p.wait()
Run Code Online (Sandbox Code Playgroud)

  • 您似乎在上一个解决方案中将所有子项的stdout复用到单个fd(output_fd).如果2个孩子同时打印怎么办,不会弄乱输出(例如'AAA \n'+'BBB \n' - >'ABBB \nAA \n') (2认同)
  • @dan3:这是一个合理的担忧。小于 `PIPE_BUF` 字节的 `write` 是原子的。否则来自多个进程的数据可能会交错。POSIX 至少需要 512 个字节。在 Linux 上,`PIPE_BUF` 是 4096 字节。 (2认同)

jfs*_*jfs 6

您还可以使用twisted以下方法同时从多个子进程中收集stdout :

#!/usr/bin/env python
import sys
from twisted.internet import protocol, reactor

class ProcessProtocol(protocol.ProcessProtocol):
    def outReceived(self, data):
        print data, # received chunk of stdout from child

    def processEnded(self, status):
        global nprocesses
        nprocesses -= 1
        if nprocesses == 0: # all processes ended
            reactor.stop()

# start subprocesses
nprocesses = 5
for _ in xrange(nprocesses):
    reactor.spawnProcess(ProcessProtocol(), sys.executable,
                         args=[sys.executable, 'child.py'],
                         usePTY=True) # can change how child buffers stdout
reactor.run()
Run Code Online (Sandbox Code Playgroud)

请参阅在Twisted中使用进程.