c/c++实现获取域名的IP地址
// GetHostIP.cpp : 定义控制台应用程序的入口点。 // #include "stdafx.h" #include#include #include #include #pragma comment(lib, "ws2_32.lib") int main(int argc, char **argv) { //----------------------------------------- // Declare and initialize variables WSADATA wsaData; int iResult; DWORD dwError; int i = 0; struct hostent *remoteHost; char *host_name; struct in_addr addr; char **pAlias; // Validate the parameters if (argc != 2) { printf("usage: GetHostIP hostnamen"); 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 %sn", 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("主机名:%sn",remoteHost->h_name); for(i=0;;i++){ if(remoteHost->h_addr_list[i]!=0) { addr.s_addr = *(u_long*)remoteHost->h_addr_list[i]; printf("ip #%d:%sn",i,inet_ntoa(addr)); } } } // 此处应添加的程序如下 // 程序完成后应当适当测试,需要进行的测试如下: // 1. 测试主机结构指针获取失败 // 2. 测试包含多个IP地址的主机 // 3. 你能想到的任何可能出现的异常情况 /////////////////结束/////////////////////////////////// system("pause"); return 0; }
下面给大家详细介绍下上面代码的核心gethostbyname的详细用法
使用这个东西,首先要包含2个头文件:
#include#include struct hostent *gethostbyname(const char *name);
这个函数的传入值是域名或者主机名,例如"www.google.com","wpc"等等。
传出值,是一个hostent的结构(如下)。如果函数调用失败,将返回NULL。
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。
下面是例程,有详细的注释。
#include#include int main(int argc, char **argv) { char *ptr,**pptr; struct hostent *hptr; char str[32]; ptr = argv[1]; if( (hptr = gethostbyname(ptr) ) == NULL ) { printf("gethostbyname error for host:%s/n", ptr); return 0; } 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; 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的代码:
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:%sn",destIP);
ip = destIP;
return true;
}
return true;
}
获取内网IP
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
}



