C++ 为什么getaddrinfo()返回IPv4地址,即使我选择了AF_INET6?
这是来自MSDN的示例代码。 我只将C++ 为什么getaddrinfo()返回IPv4地址,即使我选择了AF_INET6?,c++,sockets,winsock2,C++,Sockets,Winsock2,这是来自MSDN的示例代码。 我只将hits.ai_family更改为AF_INET6 int main() { WSADATA wsaData; int iResult; // Initialize Winsock iResult = WSAStartup(MAKEWORD(2, 2), &wsaData); if (iResult != 0) { printf("WSAStartup failed: %d\n", iRe
hits.ai_family
更改为AF_INET6
int main()
{
WSADATA wsaData;
int iResult;
// Initialize Winsock
iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);
if (iResult != 0) {
printf("WSAStartup failed: %d\n", iResult);
return 1;
}
ZeroMemory(&hints, sizeof(hints));
hints.ai_family = AF_INET6;
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();
return 1;
}
ptr = result;
SOCKET ListenSocket = INVALID_SOCKET;
ListenSocket = socket(result->ai_family, result->ai_socktype, result->ai_protocol);
if (ListenSocket == INVALID_SOCKET) {
printf("Error at socket(): %ld\n", WSAGetLastError());
freeaddrinfo(result);
WSACleanup();
return 1;
}
// Setup the TCP listening socket
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();
return 1;
}
freeaddrinfo(result);
if (listen(ListenSocket, SOMAXCONN) == SOCKET_ERROR) {
printf("Listen failed with error: %ld\n", WSAGetLastError());
closesocket(ListenSocket);
WSACleanup();
return 1;
}
SOCKET ClientSocket;
ClientSocket = INVALID_SOCKET;
// Accept a client socket
ClientSocket = accept(ListenSocket, NULL, NULL);
if (ClientSocket == INVALID_SOCKET) {
printf("accept failed: %d\n", WSAGetLastError());
closesocket(ListenSocket);
WSACleanup();
return 1;
}
}
当我检查返回的地址时,result->ai_addr
是一个4字节的十六进制数。如何使其返回IPv6地址
编辑:由于我(也)认为我可能遗漏了一些要点,下图显示了我认为它的IP地址
找到了解决办法
struct sockaddr_in6* ipv6_address = (struct sockaddr_in6*)&(result->ai_addr);
我正在铸造ai_addr的指针地址。更正后,我用
inet\u ntop
检查了它,现在它工作了。你怎么知道它返回了8个字节?(IPv4将是4个字节。)您无法通过查看原始的sockaddr
结构来确定任何内容。您应该检查result->ai_addr->sa_family
,然后将其转换为一个指针,指向适合给定家族的结构(例如,IPv4的sockaddr_in,IPv6的sockaddr_in 6
。@IgorTandetnik您能给出一个简单的示例代码吗?(仅转换部分)我不知道如何强制转换这些不同的结构。sockaddr\u in*ipv4\u address=(sockaddr\u in*)(result->ai\u addr)代码>转换为sockaddr_in6
留给读者作为练习。不管怎样-您是否检查了result->ai_addr->sa_family
?它有什么价值?回答这个问题不需要施法。@IgorTandetnik施法后,我仍然可以看到4字节的地址。也许我遗漏了什么。你怎么知道它返回了8字节?(IPv4将是4个字节。)您无法通过查看原始的sockaddr
结构来确定任何内容。您应该检查result->ai_addr->sa_family
,然后将其转换为一个指针,指向适合给定家族的结构(例如,IPv4的sockaddr_in,IPv6的sockaddr_in 6
。@IgorTandetnik您能给出一个简单的示例代码吗?(仅转换部分)我不知道如何强制转换这些不同的结构。sockaddr\u in*ipv4\u address=(sockaddr\u in*)(result->ai\u addr)代码>转换为sockaddr_in6
留给读者作为练习。不管怎样-您是否检查了result->ai_addr->sa_family
?它有什么价值?回答这个问题不需要施法。@IgorTandetnik施法后,我仍然可以看到4字节的地址。也许我遗漏了一些要点。