欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

c/c++實現(xiàn)獲取域名的IP地址

 更新時間:2015年11月08日 16:00:58   投稿:hebedich  
本文給大家匯總介紹了使用c/c++實現(xiàn)獲取域名的IP地址的幾種方法以及這些方法的核心函數(shù)gethostbyname的詳細用法,非常的實用,有需要的小伙伴可以參考下。

c/c++實現(xiàn)獲取域名的IP地址

// GetHostIP.cpp : 定義控制臺應(yīng)用程序的入口點。
//
 
#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結(jié)構(gòu)被用來儲存調(diào)用AfxSocketInit全局函數(shù)返回的Windows Sockets初始化信息。
  *  這個結(jié)構(gòu)被用來存儲被WSAStartup函數(shù)調(diào)用后返回的Windows Sockets數(shù)據(jù)。它包含Winsock.dll執(zhí)行的數(shù)據(jù)。
  */
  WSADATA wsaData;
  int iResult;
 
  DWORD dwError; /*每個word為2個字節(jié)的長度,DWORD 雙字即為4個字節(jié),每個字節(jié)是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;
  }
  // 此處應(yīng)添加的程序如下
  // 1. 當(dāng)初始化winsocket
  iResult = WSAStartup(MAKEWORD(2,2),&wsaData);/* & 取地址*/
  // 2. 檢查該socket是否初始化成功,即該socket是否等于0;如果初始化不成功,應(yīng)當(dāng)給出錯誤報警,并結(jié)束程序。
  if(iResult!=0){
    printf("初始化失敗!\n");
    return 1;
  }
 
  /////////////////結(jié)束///////////////////////////////////
 
 
  host_name = argv[1];
  printf("Calling gethostbyname with %s\n", host_name);
  // 此處應(yīng)添加的程序如下
  // 1. 利用函數(shù)gethostbyname(),獲取給定主機名的指針。
  remoteHost = gethostbyname(host_name);
  // 2. 應(yīng)當(dāng)熟悉該結(jié)構(gòu)指針的結(jié)構(gòu)
  // 其中該函數(shù)申明如下:struct hostent* gethostbyname(const char *name)
 
   
  // 此處應(yīng)添加的程序如下
  // 1. 如果上面函數(shù)返回的主機結(jié)構(gòu)指針為空(NULL),則做如下處理:
  //   a. 利用函數(shù) int WSAGetLastError ( void ) 檢查當(dāng)前是否發(fā)生網(wǎng)絡(luò)錯誤,
  //   b. 返回的發(fā)生的錯誤類型并作相應(yīng)的處理,比如,若沒有找到主機的錯誤(此時該函數(shù)返回WSAHOST_NOT_FOUND)
  if(remoteHost == NULL){
    //printf("gethostbynameError:%d",WSAGetLastError());
    return 1;
  }else{
  // 2. 如果返回的主機指針不為空,則做如下處理:
  //   a. 打印出如下參數(shù):主機名和IP地址,若該主機對應(yīng)于多個ip地址,應(yīng)當(dāng)分別列出。
    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地址轉(zhuǎn)化為0xXXXXXXXX形式的地址格式。
      */
        addr.s_addr = *(u_long*)remoteHost->h_addr_list[i]; 
        printf("ip #%d:%s\n",i,inet_ntoa(addr)); /* inet_ntoa() 函數(shù)將網(wǎng)絡(luò)地址轉(zhuǎn)成二進制的數(shù)字相關(guān)函數(shù):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地址轉(zhuǎn)化為0xXXXXXXXX形式的地址格式。
      *
      memcpy(&addr.S_un.S_addr,p,remoteHost->h_length); 
      printf("ip地址為:%s\n",inet_ntoa(addr));
     
    */
     
    }
     
  }
   
 
 
  // 此處應(yīng)添加的程序如下
  // 程序完成后應(yīng)當(dāng)適當(dāng)測試,需要進行的測試如下:
  // 1. 測試主機結(jié)構(gòu)指針獲取失敗
  // 2. 測試包含多個IP地址的主機
  // 3. 你能想到的任何可能出現(xiàn)的異常情況
  /////////////////結(jié)束///////////////////////////////////
   
   
  system("pause"); /*防止窗體關(guān)閉函數(shù)*/
  return 0;
}

下面給大家詳細介紹下上面代碼的核心gethostbyname的詳細用法

使用這個東西,首先要包含2個頭文件:

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

這個函數(shù)的傳入值是域名或者主機名,例如" 傳出值,是一個hostent的結(jié)構(gòu)(如下)。如果函數(shù)調(diào)用失敗,將返回NULL。

struct hostent {
   char *h_name;
   char **h_aliases;
   int h_addrtype;
   int h_length;
   char **h_addr_list;
};

解釋一下這個結(jié)構(gòu), 其中:

char *h_name 表示的是主機的規(guī)范名。例如www.google.com的規(guī)范名其實是www.l.google.com。
char **h_aliases 表示的是主機的別名。www.google.com就是google他自己的別名。有的時候,有的主機可能有好幾個別名,這些,其實都是為了易于用戶記憶而為自己的網(wǎng)站多取的名字。
int h_addrtype 表示的是主機ip地址的類型,到底是ipv4(AF_INET),還是ipv6(AF_INET6)
int h_length 表示的是主機ip地址的長度
int **h_addr_lisst 表示的是主機的ip地址,注意,這個是以網(wǎng)絡(luò)字節(jié)序存儲的。千萬不要直接用printf帶%s參數(shù)來打這個東西,會有問題的哇。所以到真正需要打印出這個IP的話,需要調(diào)用inet_ntop()。
const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt) :
這個函數(shù),是將類型為af的網(wǎng)絡(luò)地址結(jié)構(gòu)src,轉(zhuǎn)換成主機序的字符串形式,存放在長度為cnt的字符串中。
這個函數(shù),其實就是返回指向dst的一個指針。如果函數(shù)調(diào)用錯誤,返回值是NULL。
下面是例程,有詳細的注釋。

#include <netdb.h>
#include <sys/socket.h>
int main(int argc, char **argv)
{
char *ptr,**pptr;
struct hostent *hptr;
char str[32];
/* 取得命令后第一個參數(shù),即要解析的域名或主機名 */
ptr = argv[1];
/* 調(diào)用gethostbyname()。調(diào)用結(jié)果都存在hptr中 */
if( (hptr = gethostbyname(ptr) ) == NULL )
{
printf("gethostbyname error for host:%s/n", ptr);
return 0; /* 如果調(diào)用gethostbyname發(fā)生錯誤,返回1 */
}
/* 將主機的規(guī)范名打出來 */
printf("official hostname:%s/n",hptr->h_name);
/* 主機可能有多個別名,將所有別名分別打出來 */
for(pptr = hptr->h_aliases; *pptr != NULL; pptr++)
printf(" alias:%s/n",*pptr);
/* 根據(jù)地址類型,將地址打出來 */
switch(hptr->h_addrtype)
{
case AF_INET:
case AF_INET6:
pptr=hptr->h_addr_list;
/* 將剛才得到的所有地址都打出來。其中調(diào)用了inet_ntop()函數(shù) */
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;
} 

另外附上獲得公網(wǎng)與內(nèi)網(wǎng)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;
}

獲取內(nèi)網(wǎng)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,繼續(xù)下一個
		{
			ifreq++;
			continue;
		}
	}
	strcpy(outip,ip);
	return 0;
#else
	return 0;
#endif
	
}

相關(guān)文章

  • C語言中的程序環(huán)境與預(yù)處理詳情

    C語言中的程序環(huán)境與預(yù)處理詳情

    這篇文章主要介紹了C語言中的程序環(huán)境與預(yù)處理詳情,文章圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的朋友可以參考一下
    2022-07-07
  • C++ 創(chuàng)建桌面快捷方式 開始菜單的實現(xiàn)代碼

    C++ 創(chuàng)建桌面快捷方式 開始菜單的實現(xiàn)代碼

    這篇文章介紹了C++ 創(chuàng)建桌面快捷方式,開始菜單的實現(xiàn)代碼,需要的朋友可以參考一下
    2013-06-06
  • C語言?程序的編譯系統(tǒng)解析

    C語言?程序的編譯系統(tǒng)解析

    編譯程序的基本功能是把源程序(高級語言)翻譯成目標程序。但是,作為一個具有實際應(yīng)用價值的編譯系統(tǒng),除了基本功能之外,還應(yīng)具備語法檢查、調(diào)試措施、修改手段、覆蓋處理、目標程序優(yōu)化、不同語言合用以及人-機聯(lián)系等重要功能
    2022-02-02
  • C++中vector<vector<int>?>的基本使用方法

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

    vector<vector<int>?>其實就是容器嵌套容器,外層容器的元素類型是vector<int>,下面這篇文章主要給大家介紹了關(guān)于C++中vector<vector<int>?>的基本使用方法,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-07-07
  • C++中的String的常用函數(shù)用法(最新推薦)

    C++中的String的常用函數(shù)用法(最新推薦)

    這篇文章主要介紹了C++中的String的常用函數(shù)用法總結(jié),本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-02-02
  • C++關(guān)于樹的定義全面梳理

    C++關(guān)于樹的定義全面梳理

    樹是一種重要的非線性數(shù)據(jù)結(jié)構(gòu),直觀地看,它是數(shù)據(jù)元素(在樹中稱為結(jié)點)按分支關(guān)系組織起來的結(jié)構(gòu),很象自然界中的樹那樣。樹結(jié)構(gòu)在客觀世界中廣泛存在,如人類社會的族譜和各種社會組織機構(gòu)都可用樹形象表示,本篇介紹二叉樹的遞歸與非遞歸遍歷的方法
    2022-06-06
  • Opencv實現(xiàn)對象提取與測量

    Opencv實現(xiàn)對象提取與測量

    這篇文章主要為大家詳細介紹了基于Opencv實現(xiàn)對象提取與測量,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • Qt基礎(chǔ)開發(fā)之QString與QByteArray詳細用法與區(qū)別及QString QByteArray互轉(zhuǎn)

    Qt基礎(chǔ)開發(fā)之QString與QByteArray詳細用法與區(qū)別及QString QByteArray互轉(zhuǎn)

    這篇文章主要介紹了Qt基礎(chǔ)開發(fā)之QString與QByteArray詳細用法與區(qū)別及QString QByteArray互轉(zhuǎn),需要的朋友可以參考下
    2020-03-03
  • C++堆排序算法的實現(xiàn)方法

    C++堆排序算法的實現(xiàn)方法

    這篇文章主要介紹了C++堆排序算法的實現(xiàn)方法,很經(jīng)典的算法,需要的朋友可以參考下
    2014-08-08
  • Qt5.9繼承QObject創(chuàng)建多線程實例

    Qt5.9繼承QObject創(chuàng)建多線程實例

    本文主要介紹了Qt5.9繼承QObject創(chuàng)建多線程實例,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09

最新評論