1

我需要使用 WinSock 从特定的本地网卡打开一个套接字。我问了这个问题并在这里得到了答案。总之,答案建议你先绑定到本地接口,然后调用connect。

但是,当我这样做时,我得到一个 WSAEADDRNOTAVAIL (10049) “请求的地址在其上下文中无效。”。为什么是这样?

假设下面的示例代码是在本地机器 192.168.1.3 上运行的应用程序的一部分,并且正在尝试连接到远程服务器 192.168.1.4。我已经检查并仔细检查了本地和远程地址是否正确。我可以 ping 两种方式(从本地到远程和远程到本地)。

我已经为本地尝试了 0 以外的端口;没有不同。如果我在连接之前删除绑定,它就会起作用,但我无法指定网络接口。

那么,知道为什么我不断收到 WSAEADDRNOTAVAIL 吗?

addrinfo localhints = {0};
localhints.ai_flags = AI_NUMERICHOST | AI_NUMERICSERV;
localhints.ai_family = AF_INET;
localhints.ai_socktype = SOCK_STREAM;
localhints.ai_protocol = IPPROTO_TCP;

addrinfo *localaddr = NULL;
getaddrinfo("192.168.1.3", "0", &localhints, &localaddr);
bind(s, localaddr->ai_addr, localaddr->ai_addrlen);
freeaddrinfo(localaddr);

addrinfo remotehints = {0};
remotehints.ai_flags = AI_NUMERICHOST | AI_NUMERICSERV;
remotehints.ai_family = AF_INET;
remotehints.ai_socktype = SOCK_STREAM;
remotehints.ai_protocol = IPPROTO_TCP;

addrinfo *remoteaddr = NULL;
getaddrinfo("192.168.1.4", "12345", &remotehints, &remoteaddr);
connect(s, remoteaddr->ai_addr, remoteaddr->ai_addrlen);
freeaddrinfo(remoteaddr);

编辑:此示例代码故意没有错误检查,以便可以以最有效的方式传达我的意图。

编辑 2:绑定到 192.168.1.3 会导致连接失败。绑定到 127.0.0.1 有效。是的,我 100% 确定 192.168.1.3 是正确的本地 IP。

编辑3:对!一时兴起,我在家用 PC 上试用了测试应用程序,它运行良好。所以,至少代码确实有效,问题肯定与我的工作电脑有关。

4

2 回答 2

1

好的,事实证明@claptrap 是对的。

我禁用了我的防火墙,但我不知道,我们的 IT 部门还有一些其他防火墙功能紧密集成到企业病毒扫描程序中(我无法卸载或禁用)。一旦我设法让 IT 暂时禁用它,一切都按预期工作。

于 2013-02-18T21:00:00.693 回答
0

调用 API 函数时始终检查错误代码,例如:

addrinfo localhints = {0};
localhints.ai_flags = AI_NUMERICHOST | AI_NUMERICSERV;
localhints.ai_family = AF_INET;
localhints.ai_socktype = SOCK_STREAM;
localhints.ai_protocol = IPPROTO_TCP;

addrinfo *localaddr = NULL;
int ret = getaddrinfo("192.168.1.3", "0", &localhints, &localaddr);
if (ret == 0)
{
    ret = bind(s, localaddr->ai_addr, localaddr->ai_addrlen);
    freeaddrinfo(localaddr);

    if (ret == 0)
    {
        addrinfo remotehints = {0};
        remotehints.ai_flags = AI_NUMERICHOST | AI_NUMERICSERV;
        remotehints.ai_family = AF_INET;
        remotehints.ai_socktype = SOCK_STREAM;
        remotehints.ai_protocol = IPPROTO_TCP;

        addrinfo *remoteaddr = NULL;
        ret = getaddrinfo("192.168.1.4", "12345", &remotehints, &remoteaddr);
        if (ret == 0)
        {
            ret = connect(s, remoteaddr->ai_addr, remoteaddr->ai_addrlen);
            freeaddrinfo(remoteaddr);

            if (ret == 0)
            {
                // connect succeeded...

                // can use getsockname() here to discover which port was actually bound to, if needed.
                // On some OS versions, bind() does not perform the actual binding immediately,
                // connect() does the actual binding instead since it can make more informed choices based on the target address being connected to...
            }
            else
            {
                // connect failed...
            }
        }
        else
        {
            // getaddrinfo() failed
        }
    }
    else
    {
        // bind failed
    }
}
else
{
    // getaddrinfo() failed...
}

代码实际上能走多远?

于 2013-02-12T11:04:22.440 回答