如何使用winsock2连接到本地主机

问题描述

我编写了此程序,以便与侦听端口27015的服务器进行内部通信。但是connect()的返回值给我错误10047。套接字创建没有问题,getaddrinfo()的返回不是'一个错误。我不知道为什么套接字无法连接,网络字节和主机字节的协调会出现问题吗?

#include <winsock2.h>
#include <windows.h>
#include <iostream>
#include <Ws2tcpip.h>
#include <iphlpapi.h>
#define DEFAULT_PORT "27015"
using namespace std;

int WINAPI WinMain (HINSTANCE hInstance,HINSTANCE hPrevInstance,PSTR szCmdLine,int iCmdshow) {
    WSADATA wsaData;
    // Initialize Winsock version 2.2
    int ret;
    // holds server info//
    if (ret = WSAStartup(MAKEWORD(2,2),&wsaData) != 0){
        std::cout << "WSAStartup Failed with error %ld\n",WSAGetLastError());
        return 0;
    }
    else{
        cout << "The current status is:" << wsaData.szSystemStatus << "\n";
    }
      
    // Setup Winsock communication code here
    struct addrinfo *result = NULL,*ptr = NULL,hints;
    hints.ai_family = AF_INET;
    hints.ai_socktype = SOCK_STREAM;
    hints.ai_protocol = IPPROTO_TCP;
       
    memset(&hints,sizeof(hints));
    int iResult = getaddrinfo("localhost",DEFAULT_PORT,&hints,&result);
    if (iResult != 0) {
        printf("getaddrinfo Failed: %d\n",iResult);
        WSACleanup();
        return 1;
    }
    SOCKET Sending_socket; 
      
    Sending_socket = socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);
    if (Sending_socket == INVALID_SOCKET){
        cout << "socket error\n";
        WSACleanup();
        return 0;
    }
     
    // Now we connect to the server,RetCode=return of connect()
    int RetCode = connect(Sending_socket,result->ai_addr,(int)result->ai_addrlen);
    if (RetCode != 0) {
        int num_attempt = 0;
        while (RetCode != 0 && num_attempt != 100){
            printf("Error at socket(): %ld\n",WSAGetLastError());
            RetCode = connect(Sending_socket,(int)result->ai_addrlen);
            num_attempt++;
        }        
    }
    else {
        cout << "connect okay \n";
    }
    int them ;
    closesocket(Sending_socket);
    // When your application is finished call WSACleanup
    them = WSACleanup();
    // making sure to cleanup
    if (them != 0){
        cout << "WSACleanup Failed with error " << WSAGetLastError();
        // and exit
        return 1;
    }
    return 1;
}

解决方法

Use getaddrinfo instead.

inet_addr function引用Microsoft文档

inet_addr函数将包含 IPv4点分十进制地址的字符串转换为IN_ADDR结构的正确地址。

强调我的。没有点分十进制地址,没有地址。

在文档页面的有关返回值的部分中也应注意

如果cp参数中的字符串不包含合法的Internet地址,例如,如果“ abcd”地址的一部分超过255,则inet_addr返回值INADDR_NONE

始终检查返回值,以确保没有给出错误消息。

,

错误10047为WSAEAFNOSUPPORT,当您尝试为connect()提供一个IP地址,该IP地址的家族不被所连接的套接字支持时,即通过传递AF_INET地址,就会发生到AF_INET6套接字,或AF_INET6地址到AF_INET套接字。在这种情况下,后者正在发生。

发生这种情况的原因是因为您对memset(&hints,sizeof(hints));的呼叫在错误的位置!

首先声明hints变量,然后不要将其清零,然后开始为AF_INET套接字填充它,然后将其清零。将memset()交给getaddrinfo()之前。因此,getaddrinfo()看到hints.ai_family的值为0(AF_UNSPEC),从而允许getaddrinfo()返回{{1}的IPv4 OR / AND IPv6地址}。 "localhost"最有可能在输出列表的最前面输出一个getaddrinfo()地址(记住,AF_INET6输出一个以空值结尾的已解析IP地址的链接列表 )。

然后,您继续使用硬编码的getaddrinfo()作为套接字系列来创建新的套接字,而不是使用AF_INET中的值,然后将result->ai_family赋予{{ 1}},如果该IP地址是result->ai_addr地址而不是connect()地址,则失败,错误10047。

话虽如此,请尝试以下类似操作:

AF_INET6