gethostbyname()
是一个用于将域名解析为 IPv4 地址的函数,它属于 C 语言的套接字编程库
使用 getaddrinfo()
代替 gethostbyname()
:
getaddrinfo()
是一个更现代、更灵活的函数,可以处理 IPv4 和 IPv6 地址。它还可以处理服务名(例如 “http”)到端口号的转换。示例代码如下:
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include<stdio.h>
int main(int argc, char *argv[]) {
struct addrinfo hints, *res;
int status;
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_UNSPEC; // AF_INET or AF_INET6 to force version
hints.ai_socktype = SOCK_STREAM;
if ((status = getaddrinfo("example.com", NULL, &hints, &res)) != 0) {
fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(status));
return 2;
}
printf("IP addresses for example.com:\n\n");
for (struct addrinfo *p = res; p != NULL; p = p->ai_next) {
void *addr;
char *ipver;
// get the pointer to the address itself,
// different fields in IPv4 and IPv6:
if (p->ai_family == AF_INET) { // IPv4
struct sockaddr_in *ipv4 = (struct sockaddr_in *)p->ai_addr;
addr = &(ipv4->sin_addr);
ipver = "IPv4";
} else { // IPv6
struct sockaddr_in6 *ipv6 = (struct sockaddr_in6 *)p->ai_addr;
addr = &(ipv6->sin6_addr);
ipver = "IPv6";
}
// convert the IP to a string and print it:
char ipstr[INET6_ADDRSTRLEN];
inet_ntop(p->ai_family, addr, ipstr, sizeof ipstr);
printf(" %s: %s\n", ipver, ipstr);
}
freeaddrinfo(res); // remember to free the memory allocated by getaddrinfo!
return 0;
}
使用线程或异步操作:
如果你希望在等待 DNS 查询完成时执行其他任务,可以考虑使用多线程或异步操作。例如,你可以使用 POSIX 线程(pthread)或 C++11 的 std::thread
在后台运行 DNS 查询。这样,你的程序可以在等待查询完成时继续执行其他任务。
缓存 DNS 查询结果: 如果你的应用程序需要多次查询相同的域名,可以考虑实现一个简单的缓存机制。这样,当你需要查询已经查询过的域名时,可以直接从缓存中获取结果,而无需再次进行 DNS 查询。这可以节省时间并减少网络流量。
使用更高级的库: 有些第三方库提供了更高级的 DNS 查询功能,例如 c-ares、libdns 等。这些库通常提供了异步查询、更好的错误处理和额外的功能。你可以根据你的需求选择一个合适的库来优化你的 DNS 查询。
请注意,这些示例代码仅适用于 Unix 系统(如 Linux 和 macOS)。如果你正在使用 Windows,你需要包含相应的头文件(如 <winsock2.h>
和 <ws2tcpip.h>
),并链接到相应的库(如 ws2_32.lib
)。此外,你可能需要调整代码以适应 Windows 的特定功能和限制。