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

 更新时间:2015年11月08日 16:00:58   投稿:hebedich  
本文给大家汇总介绍了使用c/c++实现获取域名的IP地址的几种方法以及这些方法的核心函数gethostbyname的详细用法,非常的实用,有需要的小伙伴可以参考下。

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

// 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个头文件:

#include <netdb.h>
#include <sys/socket.h>
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 <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的代码:

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

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
	
}

相关文章

  • C语言中的程序环境与预处理详情

    C语言中的程序环境与预处理详情

    这篇文章主要介绍了C语言中的程序环境与预处理详情,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的朋友可以参考一下
    2022-07-07
  • C++ 创建桌面快捷方式 开始菜单的实现代码

    C++ 创建桌面快捷方式 开始菜单的实现代码

    这篇文章介绍了C++ 创建桌面快捷方式,开始菜单的实现代码,需要的朋友可以参考一下
    2013-06-06
  • C语言 程序的编译系统解析

    C语言 程序的编译系统解析

    编译程序的基本功能是把源程序(高级语言)翻译成目标程序。但是,作为一个具有实际应用价值的编译系统,除了基本功能之外,还应具备语法检查、调试措施、修改手段、覆盖处理、目标程序优化、不同语言合用以及人-机联系等重要功能
    2022-02-02
  • C++中vector<vector<int> >的基本使用方法

    C++中vector<vector<int> >的基本使用方法

    vector<vector<int> >其实就是容器嵌套容器,外层容器的元素类型是vector<int>,下面这篇文章主要给大家介绍了关于C++中vector<vector<int> >的基本使用方法,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-07-07
  • C++中的String的常用函数用法(最新推荐)

    C++中的String的常用函数用法(最新推荐)

    这篇文章主要介绍了C++中的String的常用函数用法总结,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-02-02
  • C++关于树的定义全面梳理

    C++关于树的定义全面梳理

    树是一种重要的非线性数据结构,直观地看,它是数据元素(在树中称为结点)按分支关系组织起来的结构,很象自然界中的树那样。树结构在客观世界中广泛存在,如人类社会的族谱和各种社会组织机构都可用树形象表示,本篇介绍二叉树的递归与非递归遍历的方法
    2022-06-06
  • Opencv实现对象提取与测量

    Opencv实现对象提取与测量

    这篇文章主要为大家详细介绍了基于Opencv实现对象提取与测量,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-05-05
  • Qt基础开发之QString与QByteArray详细用法与区别及QString QByteArray互转

    Qt基础开发之QString与QByteArray详细用法与区别及QString QByteArray互转

    这篇文章主要介绍了Qt基础开发之QString与QByteArray详细用法与区别及QString QByteArray互转,需要的朋友可以参考下
    2020-03-03
  • C++堆排序算法的实现方法

    C++堆排序算法的实现方法

    这篇文章主要介绍了C++堆排序算法的实现方法,很经典的算法,需要的朋友可以参考下
    2014-08-08
  • Qt5.9继承QObject创建多线程实例

    Qt5.9继承QObject创建多线程实例

    本文主要介绍了Qt5.9继承QObject创建多线程实例,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09

最新评论