首页 > 编程 > C++ > 正文

c/c++实现获取域名的IP地址

2020-05-23 14:13:09
字体:
来源:转载
供稿:网友

本文给大家汇总介绍了使用c/c++实现获取域名的IP地址的几种方法以及这些方法的核心函数gethostbyname的详细用法,非常的实用,有需要的小伙伴可以参考下。

c/c++实现获取域名的IP地址

 

 
  1. // GetHostIP.cpp : 定义控制台应用程序的入口点。 
  2. // 
  3.  
  4. #include "stdafx.h" 
  5.  
  6. #include <winsock2.h> 
  7. #include <ws2tcpip.h> 
  8. #include <stdio.h> 
  9. #include <windows.h> 
  10. #pragma comment(lib, "ws2_32.lib") 
  11.  
  12. int main(int argc, char **argv) 
  13.  
  14. //----------------------------------------- 
  15. // Declare and initialize variables 
  16. /* 
  17. * WSADATA结构被用来储存调用AfxSocketInit全局函数返回的Windows Sockets初始化信息。 
  18. * 这个结构被用来存储被WSAStartup函数调用后返回的Windows Sockets数据。它包含Winsock.dll执行的数据。 
  19. */ 
  20. WSADATA wsaData; 
  21. int iResult; 
  22.  
  23. DWORD dwError; /*每个word为2个字节的长度,DWORD 双字即为4个字节,每个字节是8位*/ 
  24. int i = 0; 
  25.  
  26. struct hostent *remoteHost; /*域名*/ 
  27. char *host_name; /*主机名*/ 
  28. struct in_addr addr; /*ip*/ 
  29.  
  30. char **pAlias; 
  31.  
  32. // Validate the parameters 
  33. if (argc != 2) { 
  34. printf("usage: GetHostIP hostname/n"); 
  35. return 1; 
  36. // 此处应添加的程序如下 
  37. // 1. 当初始化winsocket 
  38. iResult = WSAStartup(MAKEWORD(2,2),&wsaData);/* & 取地址*/ 
  39. // 2. 检查该socket是否初始化成功,即该socket是否等于0;如果初始化不成功,应当给出错误报警,并结束程序。 
  40. if(iResult!=0){ 
  41. printf("初始化失败!/n"); 
  42. return 1; 
  43.  
  44. /////////////////结束/////////////////////////////////// 
  45.  
  46.  
  47. host_name = argv[1]; 
  48. printf("Calling gethostbyname with %s/n", host_name); 
  49. // 此处应添加的程序如下 
  50. // 1. 利用函数gethostbyname(),获取给定主机名的指针。 
  51. remoteHost = gethostbyname(host_name); 
  52. // 2. 应当熟悉该结构指针的结构 
  53. // 其中该函数申明如下:struct hostent* gethostbyname(const char *name) 
  54.  
  55.  
  56. // 此处应添加的程序如下 
  57. // 1. 如果上面函数返回的主机结构指针为空(NULL),则做如下处理: 
  58. // a. 利用函数 int WSAGetLastError ( void ) 检查当前是否发生网络错误, 
  59. // b. 返回的发生的错误类型并作相应的处理,比如,若没有找到主机的错误(此时该函数返回WSAHOST_NOT_FOUND) 
  60. if(remoteHost == NULL){ 
  61. //printf("gethostbynameError:%d",WSAGetLastError()); 
  62. return 1; 
  63. }else
  64. // 2. 如果返回的主机指针不为空,则做如下处理: 
  65. // a. 打印出如下参数:主机名和IP地址,若该主机对应于多个ip地址,应当分别列出。 
  66. printf("主机名:%s/n",remoteHost->h_name); 
  67. for(i=0;;i++){ 
  68. if(remoteHost->h_addr_list[i]!=0) 
  69. /*从缓存中把 p 拷贝到addr中 
  70. *同时addr.S_un.Saddr 
  71. * in_addr ipAddr;  
  72. * ipAddr.S_un.S_addr = inet_addr("127.0.0.1");  
  73. * 就是把字符串形式的ip地址转化为0xXXXXXXXX形式的地址格式。 
  74. */ 
  75. addr.s_addr = *(u_long*)remoteHost->h_addr_list[i];  
  76. printf("ip #%d:%s/n",i,inet_ntoa(addr)); /* inet_ntoa() 函数将网络地址转成二进制的数字相关函数:inet_aton, inet_ntoa */ 
  77. /* 
  78. for(i=0;;i++){ 
  79. char *p = remoteHost->h_addr_list[i]; 
  80. if(p==NULL) break; 
  81. /*从缓存中把 p 拷贝到addr中 
  82. *同时addr.S_un.Saddr 
  83. * in_addr ipAddr;  
  84. * ipAddr.S_un.S_addr = inet_addr("127.0.0.1");  
  85. * 就是把字符串形式的ip地址转化为0xXXXXXXXX形式的地址格式。 
  86. * 
  87. memcpy(&addr.S_un.S_addr,p,remoteHost->h_length);  
  88. printf("ip地址为:%s/n",inet_ntoa(addr)); 
  89.  
  90. */ 
  91.  
  92.  
  93.  
  94.  
  95.  
  96. // 此处应添加的程序如下 
  97. // 程序完成后应当适当测试,需要进行的测试如下: 
  98. // 1. 测试主机结构指针获取失败 
  99. // 2. 测试包含多个IP地址的主机 
  100. // 3. 你能想到的任何可能出现的异常情况 
  101. /////////////////结束/////////////////////////////////// 
  102.  
  103.  
  104. system("pause"); /*防止窗体关闭函数*/ 
  105. return 0; 

下面给大家详细介绍下上面代码的核心gethostbyname的详细用法

使用这个东西,首先要包含2个头文件:

 

 
  1. #include <netdb.h> 
  2. #include <sys/socket.h> 
  3. struct hostent *gethostbyname(const char *name); 

这个函数的传入值是域名或者主机名,例如"www.google.com","wpc"等等。

传出值,是一个hostent的结构(如下)。如果函数调用失败,将返回NULL。

 

 
  1. struct hostent { 
  2. char *h_name; 
  3. char **h_aliases; 
  4. int h_addrtype; 
  5. int h_length; 
  6. char **h_addr_list; 
  7. }; 

解释一下这个结构, 其中:

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. #include <netdb.h> 
  2. #include <sys/socket.h> 
  3. int main(int argc, char **argv) 
  4. char *ptr,**pptr; 
  5. struct hostent *hptr; 
  6. char str[32]; 
  7. /* 取得命令后第一个参数,即要解析的域名或主机名 */ 
  8. ptr = argv[1]; 
  9. /* 调用gethostbyname()。调用结果都存在hptr中 */ 
  10. if( (hptr = gethostbyname(ptr) ) == NULL ) 
  11. printf("gethostbyname error for host:%s/n", ptr); 
  12. return 0; /* 如果调用gethostbyname发生错误,返回1 */ 
  13. /* 将主机的规范名打出来 */ 
  14. printf("official hostname:%s/n",hptr->h_name); 
  15. /* 主机可能有多个别名,将所有别名分别打出来 */ 
  16. for(pptr = hptr->h_aliases; *pptr != NULL; pptr++) 
  17. printf(" alias:%s/n",*pptr); 
  18. /* 根据地址类型,将地址打出来 */ 
  19. switch(hptr->h_addrtype) 
  20. case AF_INET: 
  21. case AF_INET6: 
  22. pptr=hptr->h_addr_list; 
  23. /* 将刚才得到的所有地址都打出来。其中调用了inet_ntop()函数 */ 
  24. for(;*pptr!=NULL;pptr++) 
  25. printf(" address:%s/n", inet_ntop(hptr->h_addrtype, *pptr, str, sizeof(str))); 
  26. break
  27. default
  28. printf("unknown address type/n"); 
  29. break
  30. return 0; 

另外附上获得公网与内网ip的代码:

 

 
  1. bool getPublicIp(string& ip) 
  2. int sock; 
  3. char **pptr = NULL; 
  4. struct sockaddr_in destAddr; 
  5. struct hostent *ptr = NULL; 
  6. char destIP[128]; 
  7.  
  8. sock = socket(AF_INET,SOCK_STREAM,0); 
  9. if( -1 == sock ){ 
  10. perror("creat socket failed"); 
  11. return false
  12. bzero((void *)&destAddr,sizeof(destAddr)); 
  13. destAddr.sin_family = AF_INET; 
  14. destAddr.sin_port = htons(80); 
  15. ptr = gethostbyname("www.ip138.com"); 
  16. if(NULL == ptr){ 
  17. perror("gethostbyname error"); 
  18. return false
  19. for(pptr=ptr->h_addr_list ; NULL != *pptr ; ++pptr){ 
  20. inet_ntop(ptr->h_addrtype,*pptr,destIP,sizeof(destIP)); 
  21. printf("addr:%s/n",destIP); 
  22. ip = destIP; 
  23. return true
  24. return true

获取内网IP

 

 
  1. int getlocalip(char* outip) 
  2. #ifndef WIN32 
  3. int i=0; 
  4. int sockfd; 
  5. struct ifconf ifconf; 
  6. char buf[512]; 
  7. struct ifreq *ifreq; 
  8. char* ip; 
  9. //初始化ifconf 
  10. ifconf.ifc_len = 512; 
  11. ifconf.ifc_buf = buf; 
  12. strcpy(outip,"127.0.0.1"); 
  13. if((sockfd = socket(AF_INET, SOCK_DGRAM, 0))<0) 
  14. return -1; 
  15. ioctl(sockfd, SIOCGIFCONF, &ifconf); //获取所有接口信息 
  16. close(sockfd); 
  17. //接下来一个一个的获取IP地址 
  18. ifreq = (struct ifreq*)buf; 
  19. for(i=(ifconf.ifc_len/sizeof(struct ifreq)); i>0; i--) 
  20. ip = inet_ntoa(((struct sockaddr_in*)&(ifreq->ifr_addr))->sin_addr); 
  21.  
  22.  
  23. if(strcmp(ip,"127.0.0.1")==0) //排除127.0.0.1,继续下一个 
  24. ifreq++; 
  25. continue
  26. strcpy(outip,ip); 
  27. return 0; 
  28. #else 
  29. return 0; 
  30. #endif 
  31.  


发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表