一尘不染

什么条件导致打开的,无阻塞的命名管道(fifo)对于读取“不可用”?

linux

情况:

new_pipe = os.open(pipe_path, os.O_RDONLY | os.O_NONBLOCK) # pipe_path points to a FIFO
data = os.read(new_pipe, 1024)

读取偶尔会出现错误号-11:资源暂时不可用。

什么时候会出现此错误?看起来非常罕见,因为常见情况会返回数据:

  • 如果没有写入器打开了管道,则返回空的str(’‘)。
  • 如果编写器打开了管道,但FIFO中没有数据,则还返回空的str(’‘)
  • 当然,如果编写者将数据放入fifo,则将读取该数据。

阅读 440

收藏
2020-06-07

共1个答案

一尘不染

系统调用POSIX规范read(强调我的):

尝试从空管道或FIFO中读取时:

  • 如果没有进程打开要写入的管道,则read()应返回0表示文件结束。

  • 如果某个进程打开了用于写入的管道,并且设置了O_NONBLOCK,则read()将返回-1并将errno设置为[EAGAIN]。

所以基本上您的第二个假设是错误的:

如果编写器打开了管道,但FIFO中没有数据,则还返回空的str(’‘)

这将违反规范,并且我无法在我的机器上重现该行为(这EAGAIN对我来说很有意义)。这不是什么大问题,但是,您可以捕获异常并重试:

import errno

def safe_read(fd, size=1024):
   ''' reads data from a pipe and returns `None` on EAGAIN '''
   try:
      return os.read(fd, size)
   except OSError, exc:
      if exc.errno == errno.EAGAIN:
         return None
      raise
2020-06-07