0

我使用winsock 编写了一个简单的客户端/服务器应用程序。服务器和客户端通过本地主机上的 TCP 端口 76567(只是我选择的一个随机数)进行连接和通信。我在三台台式机上测试过,两台运行 XP,另一台运行 Win7,我还在四台笔记本电脑上测试过,三台运行 Win7,一台运行 XP。该应用程序在所有台式机和 XP 笔记本电脑上都可以正常工作,但在所有三台 Win7 笔记本电脑上,当客户端尝试连接到服务器时,我都会收到错误 10061!

我已经关闭了防火墙,但问题仍然存在,我还环顾四周,看看是什么导致了这个错误,看起来客户端正在尝试连接到非监听服务器。但是,服务器调用 listen() 成功返回!很奇怪,这个问题似乎只发生在 Win7 笔记本电脑上,有什么想法吗?

这是我的套接字初始化代码:

// Initialise Winsock
iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);
if(iResult != 0)
{
    printf("WSAStartup failed: %d\n", iResult);
}

// Create a server socket
ZeroMemory(&hints, sizeof(hints));
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
hints.ai_protocol = IPPROTO_TCP;
hints.ai_flags = AI_PASSIVE;

iResult = getaddrinfo(NULL, DEFAULT_PORT, &hints, &result);
if(iResult != 0)
{
    printf("getaddrinfo failed: %d\n", iResult);
    WSACleanup();
}

// Create a socket to listen for clients
listenSocket = socket(result->ai_family, result->ai_socktype, result->ai_protocol);
if(listenSocket == INVALID_SOCKET)
{
    printf("Error at socket(): %d\n", WSAGetLastError());

    freeaddrinfo(result);
    WSACleanup();
}

// Bind socket to ip address and port
iResult = bind(listenSocket, result->ai_addr, (int) result->ai_addrlen);
if(iResult == SOCKET_ERROR)
{
    printf("bind failed with error: %d\n", WSAGetLastError());

    freeaddrinfo(result);
    closesocket(listenSocket);
    WSACleanup();
}
freeaddrinfo(result);

// Listen for connection requests
if(listen(listenSocket, SOMAXCONN) != 0)
{
    printf("Listen failed with error: %d\n", WSAGetLastError());

    closesocket(listenSocket);
    WSACleanup();
}

非常感谢 :)

4

2 回答 2

2

IP 端口为 16 位整数,因此允许的最大端口号为 0xFFFF (65535)。这里发生的是一种整数溢出。由于您所需的端口号 (76567) 不适合 16 位,因此该数字被截断,仅使用最低 16 位。这为您提供了端口号 11031。该行addr.sin_port = htons(76567);应该给您一个编译器警告,因为参数htons()不能 fin into uint16_t

于 2012-07-13T21:39:25.087 回答
0

getaddrinfo()返回给定hints条件的所有可用地址的链接列表。即使机器只有一个网络适配器,它也可以分配多个 IP 地址,即使是 localhost。您正在将服务器套接字绑定到找到的第一个 IP/端口对getaddrinfo(),因此客户端可能会尝试连接到实际上没有在服务器上侦听的不同 IP/端口,例如,如果服务器绑定到您的 LAN/Internet IP,但客户端正在连接127.0.0.1127.0.0.1除非服务器绑定到,否则客户端无法连接127.0.0.1

在多宿主/多 IP 环境中,您应该在调用时使用通配符0.0.0.0IP(又名) ,而不是. 这会将套接字绑定到所有已安装网络适配器的所有可用 IP。这样,客户端可以连接到服务器绑定的任何 IP,包括. 实际上,鉴于您显示的代码,您甚至根本不需要使用:INADDR_ANYbind()result->ai_addr127.0.0.1getaddrinfo()

// Initialise Winsock
iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);
if(iResult != 0)
{
    printf("WSAStartup failed: %d\n", iResult);
}

// Create an IPv4 server socket to listen for IPv4 clients
listenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if(listenSocket == INVALID_SOCKET)
{
    printf("Error at socket(): %d\n", WSAGetLastError());
    WSACleanup();
}

// Bind socket to IPv4 address and port
sockaddr_in addr;
memset(&addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(76567);
addr.sin_addr.s_addr = INADDR_ANY;

iResult = bind(listenSocket, (sockaddr*)&addr, sizeof(addr));
if(iResult == SOCKET_ERROR)
{
    printf("bind failed with error: %d\n", WSAGetLastError());
    closesocket(listenSocket);
    WSACleanup();
}

// Listen for IPv4 connection requests
if(listen(listenSocket, SOMAXCONN) != 0)
{
    printf("Listen failed with error: %d\n", WSAGetLastError());
    closesocket(listenSocket);
    WSACleanup();
}
于 2012-04-28T00:41:33.880 回答