ERROR_INVALID_HANDLE 调用 ConnectNamedPipe 时

ERROR_INVALID_HANDLE when calling ConnectNamedPipe

我正在尝试实现一个系统,其中任意数量的进程将通过命名管道相互通信。 Windows 中的 C++ 实现。但是,我被卡住了,因为进程似乎无法接收消息。我找不到类似的问题,如果已经有人问过,我深表歉意。无论如何,这是与句柄创建相关的代码。我的想法是我有 n(n=大约 5 或 6)个进程,每个进程都有一个用于写入(pipesIn)和读取(pipesOut)的命名管道。

pipesIn = std::vector<HANDLE *>();
    pipesOut = std::vector<HANDLE *>();
    this->name = name;
    wchar_t pipeName[20];
    for (int i = 0; i < total; i++) {
        if (i != name) {
            HANDLE hIn;
            swprintf(pipeName, L"\\.\pipe\Pipe(%d,%d)", name, i);
            do {
                hIn = CreateNamedPipe(pipeName,
                    PIPE_ACCESS_DUPLEX | PIPE_TYPE_BYTE | PIPE_READMODE_BYTE,   // FILE_FLAG_FIRST_PIPE_INSTANCE is not needed but forces CreateNamedPipe(..) to fail if the pipe already exists...
                    PIPE_WAIT,
                    1,
                    1024 * 16,
                    1024 * 16,
                    NMPWAIT_USE_DEFAULT_WAIT,
                    NULL);
            } while (hIn == INVALID_HANDLE_VALUE);
            pipesIn.push_back(&hIn);
        }
    }

    for (int i = 0; i < total; i++) {
        if (i != name) {
            HANDLE hOut;
            swprintf(pipeName, L"\\.\pipe\Pipe(%d,%d)", i, name);
            do {
                hOut = CreateFile(pipeName,
                    GENERIC_READ | GENERIC_WRITE,
                    0,
                    NULL,
                    OPEN_EXISTING,
                    0,
                    NULL);
            } while (hOut == INVALID_HANDLE_VALUE);
            pipesOut.push_back(&hOut);
        }
        else {
            pipesIn.push_back(NULL);
            pipesOut.push_back(NULL);
        }
    }

我的问题出在我的接收函数中。当我调用 ConnectNamedPipe 时,结果为 0,并且 GetLastError() returns ERROR_INVALID_HANDLE.

char response[20];
    DWORD dwRead;

    bool recieved = false;
    while (!recieved) {
        bool flag = false;
        for (int i = 0; i < pipesIn.size(); i++) {
            if (name == i) {
                continue;
            }
            if (ConnectNamedPipe(pipesIn.at(i), NULL) != FALSE) {
                while (ReadFile(*(pipesIn.at(i)), response, sizeof(response) - 1, &dwRead, NULL) != FALSE)
                {
                    response[dwRead] = '[=12=]';
                    recieved = true;
                    break;
                }
            }
            else {
                printf("%d\n", GetLastError());
            }
        }
    }

由于我是 C++ 的新手并且使用管道,所以这对我来说意义不大。我试图复制示例 here 的通信方法。任何类型的建议都将被视为有用。

你把事情不必要地复杂化,搬起石头砸自己的脚。

        pipesIn.push_back(&hIn);

这是你的问题。 hIn 是一个堆栈局部变量,它在范围的末尾消失,但您正在获取堆栈上该位置的地址并保存它。

解决方法是让你的向量成为 std::vector<HANDLE>;首先,HANDLE 是非常小的对象,与指针大小相同,因此按地址获取它们不会给自己带来任何好处。