提问人:bsdfish 提问时间:3/6/2009 最后编辑:bsdfish 更新时间:3/6/2009 访问量:4828
Python Popen、关闭流和多个进程
Python Popen, closing streams and multiple processes
问:
我有一些数据,我想对 gzip、uuencode 进行编码,然后打印成标准输出。我基本上拥有的是:
compressor = Popen("gzip", stdin = subprocess.PIPE, stdout = subprocess.PIPE)
encoder = Popen(["uuencode", "dummy"], stdin = compressor.stdout)
我向压缩器提供数据的方式是通过 compressor.stdin.write(stuff)。
我真正需要做的是向压缩机发送 EOF,但我不知道该怎么做。
在某些时候,我尝试了 compressor.stdin.close(),但这不起作用——当压缩器直接写入文件时,它工作得很好,但在上述情况下,该进程不会终止并在 compressor.wait() 上停止。
建议?在这种情况下,gzip 就是一个例子,我真的需要做一些事情,将一个进程的输出通过管道传递到另一个进程。
注意:我需要压缩的数据不适合内存,因此通信在这里并不是一个好的选择。另外,如果我只是跑
compressor.communicate("Testing")
在上面的 2 行之后,它仍然挂起并出现错误
File "/usr/lib/python2.4/subprocess.py", line 1041, in communicate rlist, wlist, xlist = select.select(read_set, write_set, [])
答:
如果您只想压缩而不需要文件包装器,请考虑使用 zlib 模块
import zlib
compressed = zlib.compress("text")
shell=True 和 unix 管道建议不起作用的任何原因?
from subprocess import *
pipes = Popen("gzip | uuencode dummy", stdin=PIPE, stdout=PIPE, shell=True)
for i in range(1, 100):
pipes.stdin.write("some data")
pipes.stdin.close()
print pipes.stdout.read()
似乎有效
这不是你应该直接在 python 中做的那种事情,关于事情如何工作存在怪癖,这使得使用 shell 来做这件事成为一个更好的主意。如果你能只使用子进程。Popen(“foo |bar“, shell=True),那就更好了。
可能发生的情况是 gzip 尚未能够输出其所有输入,并且该进程在其 stdout 写入完成之前不会退出。
如果使用 strace,则可以查看进程阻止的系统调用。用于发现哪个进程是 gzip 进程,然后用于查看它正在执行的系统调用。请注意,stdin 是 FD 0,stdout 是 FD 1,您可能会看到它在这些文件描述符上读取或写入。ps auxwf
strace -p $pidnum
我怀疑问题出在你打开管道的顺序上。UUEncode 有趣的是,如果没有以正确的方式传入管道,它会在启动它时发出呜呜声(尝试在 Popen 调用中自行启动的东西,以仅使用 PIPE 作为 stdin 和 stdout 来查看爆炸)
试试这个:
encoder = Popen(["uuencode", "dummy"], stdin=PIPE, stdout=PIPE)
compressor = Popen("gzip", stdin=PIPE, stdout=encoder.stdin)
compressor.communicate("UUencode me please")
encoded_text = encoder.communicate()[0]
print encoded_text
begin 644 dummy
F'XL(`%]^L$D``PL-3<U+SD])5<A-52C(24TL3@4`;2O+"!(`````
`
end
你是对的,顺便说一句......没有办法将通用 EOF 发送到管道中。毕竟,每个程序都真正定义了自己的EOF。这样做的方法是关闭管道,就像你试图做的那样。
编辑:我应该更清楚uuencode。作为 shell 程序,它的默认行为是期望控制台输入。如果在没有“实时”传入管道的情况下运行它,它将阻止等待控制台输入。通过第二次打开编码器,在您将材料发送到压缩机管道之前,编码器正在阻塞,等待您开始打字。耶路伯是对的,因为有什么东西阻碍了。
评论