NamedPipeClientStream.Connect()抛出System.IO.FileNotFoundException:无法找到指定的文件

时间:2014-05-02 16:18:40

标签: c#

有人可以告诉我为什么NamedPipeClientStream.Connect()会抛出System.IO.FileNotFoundException异常(无法找到指定的文件)? 我认为Connect()方法将永远阻止而不是抛出异常。

提前谢谢!

1 个答案:

答案 0 :(得分:0)

NamedPipeClientStream.Connect()只是WaitNamedPipeCreateFile内核方法的托管包装器。

更多信息: https://msdn.microsoft.com/en-us/library/windows/desktop/aa365800(v=vs.85).aspx https://msdn.microsoft.com/en-us/library/windows/desktop/aa363858(v=vs.85).aspx

托管部分看起来像(简化):

public void Connect(int timeout = -1) {
  int tickCount = Environment.TickCount;
  int elapsed = 0;
  SafePipeHandle safePipeHandle;
  while (true) {
    if (!WaitNamedPipe(normalizedPipePath, timeout - elapsed)) {
      int lastWin32Error = Marshal.GetLastWin32Error();
      if (lastWin32Error != FILE_NOT_FOUND && lastWin32Error != SUCCESS) {
        ThrowWinIOError(lastWin32Error, string.Empty);
      }
    }
    if (timeout != -1 && (elapsed = Environment.TickCount - tickCount) >= timeout) {
      throw new TimeoutException();
    }

    safePipeHandle = CreateFile(normalizedPipePath, m_access, FileShare.None, null, FileMode.Open, num, null);
    if (!safePipeHandle.IsInvalid) {
      // success
      return;
    }
    int lastWin32Error2 = Marshal.GetLastWin32Error();
    if (lastWin32Error2 == PIPE_BUSY) {
      continue;
    }
    ThrowWinIOError(lastWin32Error2, string.Empty);
  }
}

因此它会调用WaitNamedPipe,如果管道不存在则会立即返回,并且托管代码会重试(如果没有指定超时,则会重新尝试)。棘手的部分是CreateFile没有处理FILE_NOT_FOUND错误(它只处理PIPE_BUSY错误) - 我怀疑这是你可以遇到种族状况的地方。

IE:WaitNamedPipe返回正常,但在CreateFile被点击之前管道发生了变化(休眠后又回到了无效状态?)并且CreateFile抛出了你看到的异常。< / p>

解决方法应该相对简单 - 用Connect方法包装来重试FILE_NOT_FOUND IOException