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 的特定功能和限制。