c/c++完成獲得域名的IP地址。本站提示廣大學習愛好者:(c/c++完成獲得域名的IP地址)文章只能為提供參考,不一定能成為您想要的結果。以下是c/c++完成獲得域名的IP地址正文
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.谷歌.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.谷歌.com的標准名實際上是www.l.谷歌.com。
char **h_aliases 表現的是主機的別號。www.谷歌.com就是谷歌他本身的別號。有的時刻,有的主機能夠有好幾個體名,這些,其實都是為了易於用戶記憶而為本身的網站多取的名字。
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
}