c/c++实现获取域名的IP地址
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
// GetHostIP.cpp : 定义控制台应用程序的入口点。 // #include "stdafx.h" #include <winsock2.h> #include <ws2tcpip.h> #include <stdio.h> #include <windows.h> #pragma comment(lib, "ws2_32.lib") int main( int argc, char **argv) { //----------------------------------------- // Declare and initialize variables /* * WSADATA结构被用来储存调用AfxSocketInit全局函数返回的Windows Sockets初始化信息。 * 这个结构被用来存储被WSAStartup函数调用后返回的Windows Sockets数据。它包含Winsock.dll执行的数据。 */ WSADATA wsaData; int iResult; DWORD dwError; /*每个word为2个字节的长度,DWORD 双字即为4个字节,每个字节是8位*/ int i = 0; struct hostent *remoteHost; /*域名*/ char *host_name; /*主机名*/ struct in_addr addr; /*ip*/ char **pAlias; // Validate the parameters if (argc != 2) { printf ( "usage: GetHostIP hostname\n" ); return 1; } // 此处应添加的程序如下 // 1. 当初始化winsocket iResult = WSAStartup(MAKEWORD(2,2),&wsaData); /* & 取地址*/ // 2. 检查该socket是否初始化成功,即该socket是否等于0;如果初始化不成功,应当给出错误报警,并结束程序。 if (iResult!=0){ printf ( "初始化失败!\n" ); return 1; } /////////////////结束/////////////////////////////////// host_name = argv[1]; printf ( "Calling gethostbyname with %s\n" , host_name); // 此处应添加的程序如下 // 1. 利用函数gethostbyname(),获取给定主机名的指针。 remoteHost = gethostbyname(host_name); // 2. 应当熟悉该结构指针的结构 // 其中该函数申明如下:struct hostent* gethostbyname(const char *name) // 此处应添加的程序如下 // 1. 如果上面函数返回的主机结构指针为空(NULL),则做如下处理: // a. 利用函数 int WSAGetLastError ( void ) 检查当前是否发生网络错误, // b. 返回的发生的错误类型并作相应的处理,比如,若没有找到主机的错误(此时该函数返回WSAHOST_NOT_FOUND) if (remoteHost == NULL){ //printf("gethostbynameError:%d",WSAGetLastError()); return 1; } else { // 2. 如果返回的主机指针不为空,则做如下处理: // a. 打印出如下参数:主机名和IP地址,若该主机对应于多个ip地址,应当分别列出。 printf ( "主机名:%s\n" ,remoteHost->h_name); for (i=0;;i++){ if (remoteHost->h_addr_list[i]!=0) { /*从缓存中把 p 拷贝到addr中 *同时addr.S_un.Saddr * in_addr ipAddr; * ipAddr.S_un.S_addr = inet_addr("127.0.0.1"); * 就是把字符串形式的ip地址转化为0xXXXXXXXX形式的地址格式。 */ addr.s_addr = *(u_long*)remoteHost->h_addr_list[i]; printf ( "ip #%d:%s\n" ,i,inet_ntoa(addr)); /* inet_ntoa() 函数将网络地址转成二进制的数字相关函数:inet_aton, inet_ntoa */ } /* for(i=0;;i++){ char *p = remoteHost->h_addr_list[i]; if(p==NULL) break; /*从缓存中把 p 拷贝到addr中 *同时addr.S_un.Saddr * in_addr ipAddr; * ipAddr.S_un.S_addr = inet_addr("127.0.0.1"); * 就是把字符串形式的ip地址转化为0xXXXXXXXX形式的地址格式。 * memcpy(&addr.S_un.S_addr,p,remoteHost->h_length); printf("ip地址为:%s\n",inet_ntoa(addr)); */ } } // 此处应添加的程序如下 // 程序完成后应当适当测试,需要进行的测试如下: // 1. 测试主机结构指针获取失败 // 2. 测试包含多个IP地址的主机 // 3. 你能想到的任何可能出现的异常情况 /////////////////结束/////////////////////////////////// system ( "pause" ); /*防止窗体关闭函数*/ return 0; } |
下面给大家详细介绍下上面代码的核心gethostbyname的详细用法
使用这个东西,首先要包含2个头文件:
1
2
3
|
#include <netdb.h> #include <sys/socket.h> struct hostent *gethostbyname( const char *name); |
这个函数的传入值是域名或者主机名,例如"www.google.com","wpc"等等。
传出值,是一个hostent的结构(如下)。如果函数调用失败,将返回NULL。
1
2
3
4
5
6
7
|
struct hostent { char *h_name; char **h_aliases; int h_addrtype; int h_length; char **h_addr_list; }; |
解释一下这个结构, 其中:
char *h_name 表示的是主机的规范名。例如www.google.com的规范名其实是www.l.google.com。
char **h_aliases 表示的是主机的别名。www.google.com就是google他自己的别名。有的时候,有的主机可能有好几个别名,这些,其实都是为了易于用户记忆而为自己的网站多取的名字。
int h_addrtype 表示的是主机ip地址的类型,到底是ipv4(AF_INET),还是ipv6(AF_INET6)
int h_length 表示的是主机ip地址的长度
int **h_addr_lisst 表示的是主机的ip地址,注意,这个是以网络字节序存储的。千万不要直接用printf带%s参数来打这个东西,会有问题的哇。所以到真正需要打印出这个IP的话,需要调用inet_ntop()。
const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt) :
这个函数,是将类型为af的网络地址结构src,转换成主机序的字符串形式,存放在长度为cnt的字符串中。
这个函数,其实就是返回指向dst的一个指针。如果函数调用错误,返回值是NULL。
下面是例程,有详细的注释。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
#include <netdb.h> #include <sys/socket.h> int main( int argc, char **argv) { char *ptr,**pptr; struct hostent *hptr; char str[32]; /* 取得命令后第一个参数,即要解析的域名或主机名 */ ptr = argv[1]; /* 调用gethostbyname()。调用结果都存在hptr中 */ if ( (hptr = gethostbyname(ptr) ) == NULL ) { printf ( "gethostbyname error for host:%s/n" , ptr); return 0; /* 如果调用gethostbyname发生错误,返回1 */ } /* 将主机的规范名打出来 */ printf ( "official hostname:%s/n" ,hptr->h_name); /* 主机可能有多个别名,将所有别名分别打出来 */ for (pptr = hptr->h_aliases; *pptr != NULL; pptr++) printf ( " alias:%s/n" ,*pptr); /* 根据地址类型,将地址打出来 */ switch (hptr->h_addrtype) { case AF_INET: case AF_INET6: pptr=hptr->h_addr_list; /* 将刚才得到的所有地址都打出来。其中调用了inet_ntop()函数 */ for (;*pptr!=NULL;pptr++) printf ( " address:%s/n" , inet_ntop(hptr->h_addrtype, *pptr, str, sizeof (str))); break ; default : printf ( "unknown address type/n" ); break ; } return 0; } |
另外附上获得公网与内网ip的代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
bool getPublicIp(string& ip) { int sock; char **pptr = NULL; struct sockaddr_in destAddr; struct hostent *ptr = NULL; char destIP[128]; sock = socket(AF_INET,SOCK_STREAM,0); if ( -1 == sock ){ perror ( "creat socket failed" ); return false ; } bzero(( void *)&destAddr, sizeof (destAddr)); destAddr.sin_family = AF_INET; destAddr.sin_port = htons(80); ptr = gethostbyname( "www.ip138.com" ); if (NULL == ptr){ perror ( "gethostbyname error" ); return false ; } for (pptr=ptr->h_addr_list ; NULL != *pptr ; ++pptr){ inet_ntop(ptr->h_addrtype,*pptr,destIP, sizeof (destIP)); printf ( "addr:%s\n" ,destIP); ip = destIP; return true ; } return true ; } |
获取内网IP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
|
int getlocalip( char * outip) { #ifndef WIN32 int i=0; int sockfd; struct ifconf ifconf; char buf[512]; struct ifreq *ifreq; char * ip; //初始化ifconf ifconf.ifc_len = 512; ifconf.ifc_buf = buf; strcpy (outip, "127.0.0.1" ); if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0))<0) { return -1; } ioctl(sockfd, SIOCGIFCONF, &ifconf); //获取所有接口信息 close(sockfd); //接下来一个一个的获取IP地址 ifreq = ( struct ifreq*)buf; for (i=(ifconf.ifc_len/ sizeof ( struct ifreq)); i>0; i--) { ip = inet_ntoa((( struct sockaddr_in*)&(ifreq->ifr_addr))->sin_addr); if ( strcmp (ip, "127.0.0.1" )==0) //排除127.0.0.1,继续下一个 { ifreq++; continue ; } } strcpy (outip,ip); return 0; #else return 0; #endif } |