Python: Using Threads To Call Subprocess.popen Multiple Times
Solution 1:
I think the key code is:
self.lock.acquire()
print"\nSubprocess started"
p = subprocess.Popen( # etc
stdout_value = proc.communicate('through stdin to stdout')[0]
self.lock.release()
the explicit calls to acquire and release should guarantee serialization -- don't you observe serialization just as invariably if you do other things in this block instead of the subprocess use?
Edit: all silence here, so I'll add the suggestion to remove the locking and instead put each stdout_value
on a Queue.Queue()
instance -- Queue is intrinsicaly threadsafe (deals with its own locking) so you can get
(or get_nowait
, etc etc) results from it once they're ready and have been put
there. In general, Queue
is the best way to arrange thread communication (and often synchronization too) in Python, any time it can be feasibly arranged to do things that way.
Specifically: add import Queue
at the start; give up making, acquiring and releasing self.lock
(just delete those three lines); add self.q = Queue.Queue()
to the __init__
; right after the call stdout_value = proc.communicate(...
add one statement self.q.put(stdout_value)
; now e.g finish the jsonrpc_run_procs
method with
while not self.q.empty():
result = self.q.get()
print'One result is %r' % result
to confirm that all the results are there. (Normally the empty
method of queues is not reliable, but in this case all threads putting to the queue are already finished, so you should be fine).
Solution 2:
Your specific problem is probably caused by the line stdout_value = proc.communicate('through stdin to stdout')[0]
. Subprocess.communicate will "Wait for process to terminate", which, when used with a lock, will run one at a time.
What you can do is simply add the p
variable to a list and run and use the Subprocess API to wait for the subprocesses to finish. Periodically poll each subprocess in your main thread.
On second look, it looks like you may have an issue on this line as well: for th in self.running_threads+self.thread_pool: th.join()
. Thread.join() is another method that will wait for the thread to finish.
Post a Comment for "Python: Using Threads To Call Subprocess.popen Multiple Times"