为什么 getaddrinfo 有多个结果?
Why are there multiple results from getaddrinfo?
我正在尝试创建一个简单的程序来获取给定特定主机名的 IP 地址:
我截取的代码附在下面:
#include <stdio.h>
#include <stdlib.h>
#include <stdio.h>
#include <netdb.h>
#include <sys/socket.h>
#include <errno.h>
#include <arpa/inet.h>
#include <string.h>
#include <unistd.h>
int main(int argc, char *argv[]) {
if(argc < 2){
printf("Please provide a hostname.\n");
exit(1);
}
char *hostname = argv[1];
char ip[100];
get_ip(hostname, ip);
printf("%s resolved to %s\n", hostname,ip);
}
int get_ip(char *hostname, char *ip){
struct sockaddr_in *h;
int sockfd;
struct addrinfo hints, *servinfo,*res;
struct addrinfo *iter;
int rv;
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
if((rv = getaddrinfo(hostname, "HTTP", &hints, &res)) != 0) {
fprintf(stderr, "getaddrinfo %s\n", gai_strerror(rv));
return 1;
}
for(iter=res; iter != NULL; iter=iter->ai_next){
printf("%p\n", iter->ai_next);
h=(struct sockaddr_in *)iter->ai_addr;
strcpy(IP, inet_ntoa(h->sin_addr));
printf("%s\n",ip);
}
freeaddrinfo(res);
return 0;
}
我输入以下参数:
gcc get_ip_addr.c -o get_ip_addr;
./get_ip_addr google-public-dns-b.google.com
这导致:
0x2475330
8.8.4.4
(nil)
0.0.0.0
当我删除 "http"
和 &hints
并将它们设置为 NULL
时,我得到以下结果:
0x1b63310
8.8.4.4
0x1b63360
8.8.4.4
0x1b633b0
8.8.4.4
0x1b63410
0.0.0.0
0x1b63470
0.0.0.0
(nil)
0.0.0.0
所以当我将 service
和 hints
设置为 NULL
时 getaddrinfo()
returns 多个可能的结果我不明白为什么我得到多个 IP 地址,而不是只获得一个 IP 地址?
通过将 service
和 hints
字段设置为 NULL
,您要求 getaddrinfo
到 return 所有可能的地址,包括 IP 和 IPv6 ,对于不同的套接字类型,你输出的只是ip地址的点表示,而这只是整个地址结构的一部分getaddrinfo
returned。如果仔细查看 returned 地址,您会发现每个地址实际上是不同的。比如下面程序一共输出6个地址:
int main(int argc, char *argv[])
{
struct addrinfo *result, *rp;
int s;
s = getaddrinfo("google-public-dns-b.google.com", NULL, NULL, &result);
if (s != 0) {
fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(s));
exit(EXIT_FAILURE);
}
char addr[1024];
for (rp = result; rp != NULL; rp = rp->ai_next) {
printf("flags: 0x%x\tfamily: %d\tsocktype: %d\tprotocol: %d\n",
rp->ai_flags,
rp->ai_family,
rp->ai_socktype,
rp->ai_protocol);
s = getnameinfo(rp->ai_addr, rp->ai_addrlen, addr, sizeof addr, NULL, 0, NI_NUMERICHOST);
if (s != 0) {
printf("getnameinfo error:%d\n", s);
continue;
}
printf("addr: %s\n", addr);
}
freeaddrinfo(result);
return 0;
}
$ ./a.out
flags: 0x28 family: 2 socktype: 1 protocol: 6
addr: 8.8.4.4
flags: 0x28 family: 2 socktype: 2 protocol: 17
addr: 8.8.4.4
flags: 0x28 family: 2 socktype: 3 protocol: 0
addr: 8.8.4.4
flags: 0x28 family: 10 socktype: 1 protocol: 6
addr: 2001:4860:4860::8844
flags: 0x28 family: 10 socktype: 2 protocol: 17
addr: 2001:4860:4860::8844
flags: 0x28 family: 10 socktype: 3 protocol: 0
addr: 2001:4860:4860::8844
如您所见,每个地址 returned 在地址、系列、套接字类型和协议的组合方面都是不同的。
如果您想 return 一种特定类型的地址,请使用 hints
将 returned 地址限制为您想要的地址。
各数据字段说明:
家庭:
#define PF_INET 2 /* IP protocol family. */
#define PF_INET6 10 /* IP version 6. */
#define AF_INET PF_INET
#define AF_INET6 PF_INET6
袜子类型:
enum __socket_type
{
SOCK_STREAM = 1, /* Sequenced, reliable, connection-based
byte streams. */
SOCK_DGRAM = 2, /* Connectionless, unreliable datagrams
of fixed maximum length. */
SOCK_RAW = 3, /* Raw protocol interface. */
....
}
协议:
enum {
IPPROTO_IP = 0, /* Dummy protocol for TCP */
IPPROTO_TCP = 6, /* Transmission Control Protocol */
IPPROTO_UDP = 17, /* User Datagram Protocol */
}
另一个问题,您在输出中看到“0.0.0.0”,因为您在 IPv6 地址上使用 inet_ntoa
而此函数仅支持 IPv4,因此它已被弃用,您应该考虑使用 inet_ntop
或 getnameinfo
支持两种地址族。
我正在尝试创建一个简单的程序来获取给定特定主机名的 IP 地址:
我截取的代码附在下面:
#include <stdio.h>
#include <stdlib.h>
#include <stdio.h>
#include <netdb.h>
#include <sys/socket.h>
#include <errno.h>
#include <arpa/inet.h>
#include <string.h>
#include <unistd.h>
int main(int argc, char *argv[]) {
if(argc < 2){
printf("Please provide a hostname.\n");
exit(1);
}
char *hostname = argv[1];
char ip[100];
get_ip(hostname, ip);
printf("%s resolved to %s\n", hostname,ip);
}
int get_ip(char *hostname, char *ip){
struct sockaddr_in *h;
int sockfd;
struct addrinfo hints, *servinfo,*res;
struct addrinfo *iter;
int rv;
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
if((rv = getaddrinfo(hostname, "HTTP", &hints, &res)) != 0) {
fprintf(stderr, "getaddrinfo %s\n", gai_strerror(rv));
return 1;
}
for(iter=res; iter != NULL; iter=iter->ai_next){
printf("%p\n", iter->ai_next);
h=(struct sockaddr_in *)iter->ai_addr;
strcpy(IP, inet_ntoa(h->sin_addr));
printf("%s\n",ip);
}
freeaddrinfo(res);
return 0;
}
我输入以下参数:
gcc get_ip_addr.c -o get_ip_addr;
./get_ip_addr google-public-dns-b.google.com
这导致:
0x2475330
8.8.4.4
(nil)
0.0.0.0
当我删除 "http"
和 &hints
并将它们设置为 NULL
时,我得到以下结果:
0x1b63310
8.8.4.4
0x1b63360
8.8.4.4
0x1b633b0
8.8.4.4
0x1b63410
0.0.0.0
0x1b63470
0.0.0.0
(nil)
0.0.0.0
所以当我将 service
和 hints
设置为 NULL
时 getaddrinfo()
returns 多个可能的结果我不明白为什么我得到多个 IP 地址,而不是只获得一个 IP 地址?
通过将 service
和 hints
字段设置为 NULL
,您要求 getaddrinfo
到 return 所有可能的地址,包括 IP 和 IPv6 ,对于不同的套接字类型,你输出的只是ip地址的点表示,而这只是整个地址结构的一部分getaddrinfo
returned。如果仔细查看 returned 地址,您会发现每个地址实际上是不同的。比如下面程序一共输出6个地址:
int main(int argc, char *argv[])
{
struct addrinfo *result, *rp;
int s;
s = getaddrinfo("google-public-dns-b.google.com", NULL, NULL, &result);
if (s != 0) {
fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(s));
exit(EXIT_FAILURE);
}
char addr[1024];
for (rp = result; rp != NULL; rp = rp->ai_next) {
printf("flags: 0x%x\tfamily: %d\tsocktype: %d\tprotocol: %d\n",
rp->ai_flags,
rp->ai_family,
rp->ai_socktype,
rp->ai_protocol);
s = getnameinfo(rp->ai_addr, rp->ai_addrlen, addr, sizeof addr, NULL, 0, NI_NUMERICHOST);
if (s != 0) {
printf("getnameinfo error:%d\n", s);
continue;
}
printf("addr: %s\n", addr);
}
freeaddrinfo(result);
return 0;
}
$ ./a.out
flags: 0x28 family: 2 socktype: 1 protocol: 6
addr: 8.8.4.4
flags: 0x28 family: 2 socktype: 2 protocol: 17
addr: 8.8.4.4
flags: 0x28 family: 2 socktype: 3 protocol: 0
addr: 8.8.4.4
flags: 0x28 family: 10 socktype: 1 protocol: 6
addr: 2001:4860:4860::8844
flags: 0x28 family: 10 socktype: 2 protocol: 17
addr: 2001:4860:4860::8844
flags: 0x28 family: 10 socktype: 3 protocol: 0
addr: 2001:4860:4860::8844
如您所见,每个地址 returned 在地址、系列、套接字类型和协议的组合方面都是不同的。
如果您想 return 一种特定类型的地址,请使用 hints
将 returned 地址限制为您想要的地址。
各数据字段说明:
家庭:
#define PF_INET 2 /* IP protocol family. */
#define PF_INET6 10 /* IP version 6. */
#define AF_INET PF_INET
#define AF_INET6 PF_INET6
袜子类型:
enum __socket_type
{
SOCK_STREAM = 1, /* Sequenced, reliable, connection-based
byte streams. */
SOCK_DGRAM = 2, /* Connectionless, unreliable datagrams
of fixed maximum length. */
SOCK_RAW = 3, /* Raw protocol interface. */
....
}
协议:
enum {
IPPROTO_IP = 0, /* Dummy protocol for TCP */
IPPROTO_TCP = 6, /* Transmission Control Protocol */
IPPROTO_UDP = 17, /* User Datagram Protocol */
}
另一个问题,您在输出中看到“0.0.0.0”,因为您在 IPv6 地址上使用 inet_ntoa
而此函数仅支持 IPv4,因此它已被弃用,您应该考虑使用 inet_ntop
或 getnameinfo
支持两种地址族。