3

刚开始研究python的管道方法。我试图将管道描述符包装到文件对象中并逐行读取。

import os,time,threading

def child():
    while True:
        time.sleep(1)
        msg = ('Spam\n' ).encode()
        os.write(pipeout,msg)


def parent(): 
    while True:
        a = os.fdopen(pipein)
        line = a.readline()[:-1]
        print('Parent %d got [%s] at %s' % (os.getpid(),line,time.time()))

pipein,pipeout = os.pipe()

threading.Thread(target=child,args=()).start()

parent()

当我运行脚本时,结果如下----脚本只在第一次迭代中工作,然后显示错误消息

Parent 621 got [Spam] at 1376785841.4  
Traceback (most recent call last):
  File "/Users/miteji/pipe-thread.py", line 43, in <module>
    parent()
  File "/Users/miteji/pipe-thread.py", line 36, in parent
    line = a.readline()[:-1]
IOError: [Errno 9] Bad file descriptor
>>> Exception in thread Thread-1:
Traceback (most recent call last):
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/threading.py",         line 551, in __bootstrap_inner
     self.run()
  File "/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/threading.py", line 504, in run
     self.__target(*self.__args, **self.__kwargs)
  File "/Users/miteji/pipe-thread.py", line 30, in child
     os.write(pipeout,msg)
OSError: [Errno 32] Broken pipe

然而,当我改变

a = os.fdopen(pipein)
line = a.readline()[:-1]

line = os.read(pipein,32)

脚本工作正常。

那么为什么不能使用“os.fdopen”方法呢?为什么管子坏了?谢谢你们!

4

1 回答 1

1

问题在于os.fdopen这里的位置:

def parent(): 
    while True:
        a = os.fdopen(pipein)
        line = a.readline()[:-1]
        print('Parent %d got [%s] at %s' % (os.getpid(),line,time.time()))

每次通过循环,您都会os.fdopen() 再次调用,即使您之前这样做过。

你第一次这样做,你没有更早做os.fdopen(),所以一切都很好。但是第二次,这重新绑定a到新结果,放弃了之前的os.fdopen()值。

当较早的值被放弃时,它就有资格进行垃圾回收。CPython 立即通知(由于引用计数)并收集它。这将删除调用os.fdclose(). 这反过来又关闭了管道。

为了解决眼前的问题,请确保您只在循环外打开管道一次。

于 2013-08-18T01:11:08.273 回答