我正在尝试创建一个简单的程序,它获取给定某个主机名的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”和&amp;提示并将它们设置为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
因此,当我将服务和提示设置为NULL时,getaddrinfo会返回多个可能的结果,我不明白为什么我得到多个ip地址而不是只获取一个IP地址? 任何帮助将不胜感激!谢谢!
答案 0 :(得分:11)
通过将service
和hints
字段设置为NULL
,您要求getaddrinfo
返回所有可能的地址,包括IP和IPv6,以及不同的套接字类型,你输出的只是ip地址的点表示,而这只是返回的整个地址结构getaddrinfo
的一部分。如果仔细查看返回的地址,您会注意到每个地址实际上是不同的。例如,以下程序输出总共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);
}
$ ./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
正如您所看到的,就地址,系列,socktype和协议的组合而言,返回的每个地址都是不同的。
如果您想返回一种特定类型的地址,请使用hints
将返回的地址限制为您想要的地址。
每个数据字段都有说明:
家族:
#define PF_INET 2 /* IP protocol family. */
#define PF_INET6 10 /* IP version 6. */
#define AF_INET PF_INET
#define AF_INET6 PF_INET6
Socktypes:
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,因此它已被弃用,你应该考虑使用{{1支持两个地址系列的{}或inet_ntop
。