学习笔记之gethostbyaddr函数

  刚才学了gethostbyname函数,这个gethostbyaddr函数的作用是通过一个IPv4的地址来获取主机信息,并放在hostent结构体中。

#include <netdb.h>
struct hostent * gethostbyaddr(const char * addr, socklen_t len, int family);//返回:若成功则为非空指针,若出错则为NULL且设置h_errno
//上面的const char * 是UNP中的写法,而我在linux 2.6中看到的是 const void *

  本函数返回一个指向hostent结构指针。addr参数实际上不是char *
类型,而是一个指向存放IPv4地址的某个in_addr结构的指针;len参数是这个结构的大小,对于IPv4地址为4;family参数为AF_INET。

  先不多说,先给出代码 (CentOS 6.4)


 1 #include <stdio.h>
2 #include <netdb.h>
3 #include <arpa/inet.h>
4 #include <sys/socket.h>
5
6 int main(int argc,char **argv)
7 {
8 char *ptr,**pptr;
9 char str[INET_ADDRSTRLEN];
10 struct hostent *hptr;
11 struct in_addr * addr;
12 struct sockaddr_in saddr;
13
14 //取参数
15 while(--argc>0)
16 {
17   ptr=*++argv; //此时的ptr是ip地址
18   if(!inet_aton(ptr,&saddr.sin_addr)) //调用inet_aton(),将ptr点分十进制转in_addr
19   {
20   printf("Inet_aton error\n");
21   return 1;
22   }
23
24   if((hptr=gethostbyaddr((void *)&saddr.sin_addr,4,AF_INET))==NULL) //把主机信息保存在hostent中
25   {
26   printf("gethostbyaddr error for addr:%s\n",ptr);
27   printf("h_errno %d\n",h_errno);
28   return 1;
29   }
30   printf("official hostname: %s\n",hptr->h_name);//正式主机名
31
32   for(pptr=hptr->h_aliases;*pptr!=NULL;pptr++)//遍历所有的主机别名
33   printf("\talias: %s\n",*pptr);
34
35   switch(hptr->h_addrtype)//判断socket类型
36   {
37   case AF_INET: //IP类为AF_INET
38   case AF_INET6: //IP类为AF_INET6
39     pptr=hptr->h_addr_list; //IP地址数组
40     for(;*pptr!=NULL;pptr++)
41     printf("\taddress: %s\n",
42     inet_ntop(hptr->h_addrtype,*pptr,str,sizeof(str)));//inet_ntop转换为点分十进制
43     break;
44   default:
45     printf("unknown address type\n");
46     break;
47   }
48 }
49 return 0;
50 }

  从代码中可以看到,gethostbyaddr的第一个参数是sockaddr_in而不是in_addr类型。我做实验的时候用in_addr作为参数,总是不行,也不知道为什么。就将就用了sockaddr_in了。

  然后我 编译后运行

  ./gethostbyaddr 127.0.0.1
 完美的成功了,正当高兴的时候。

  ./gethostbyaddr 115.239.211.110
(百度域名的ip)

  竟然出错了,是gethostbyaddr error for
addr:115.239.211.110 而h_errno是为2的。

  就找到了这一篇文章:https://community.oracle.com/thread/1926589?start=0&tstart=0

  我改了 /etc/resolv.conf 增加了一个 nameserver
8.8.8.8

  再次运行,又错了,这次的错误代码是h_errno=1。

  于是就又找到了这一篇文章:http://kb.zmanda.com/article.php?id=139

  什么,竟然要手动在/etc/hosts下增加?算了,就先试一下。写上
115.239.211.110 www.baidu.com ,运行,成功了。

  ---------------------------------------------------------------------------

  正在想为什么会这样的时候,看到UNP里面的一句话:
按照DNS的说法,gethostbyaddr在in_addr.arpa域中向一个名字服务器查询PTR记录。

  可能是我的电脑不是服务器吧,没有域名解析服务吧。所以不行。而本地的/etc/hosts差不多就是有这个功能。我就在想为什么gethostbyname会向/etc/hosts文件中查看信息,然后没有对应的话,就会返回上一级的DNS进行解析。而反向解析为什么不会自动解析呢?(Ps我想会不会是反向解析比较少用到,而且正向解析域名有层次关系,而IP没有层次关系,不方便处理吧。)我通过nslookup
115.239.211.110 进行查询时提示这个错误:

** server can‘t find 110.211.239.115.in-addr.arpa.: NXDOMAIN

  好了,没错了,要使用这个函数,本地要有反向解析的服务。

  本文地址:http://www.cnblogs.com/wunaozai/p/3753731.html

学习笔记之gethostbyaddr函数,布布扣,bubuko.com

时间: 2024-10-18 07:12:05

学习笔记之gethostbyaddr函数的相关文章

C++学习笔记--从虚函数说开去

虚函数与纯虚函数: 虚函数:在某基类中声明为virtual并在一个或多个派生类中被重新定义的成员函数,virtual  函数返回类型  函数名(参数表){函数体;} ,实现多态性,通过指向派生类的基类指针或引用,访问派生类中同名覆盖成员函数.注意虚函数在基类中是有定义的,即便定义是空. 纯虚函数:在基类中是没有定义的,必须由派生类重定义实现,否则不能由对象进行调用. 看下面的例子: #include<iostream> using namespace std; class Cshape { p

十四、Android学习笔记_Android回调函数触发的几种方式 广播 静态对象

一.通过广播方式: 1.比如登录.假如下面这个方法是外界调用的,那么怎样在LoginActivity里面执行登录操作,成功之后在回调listener接口呢?如果是平常的类,可以通过构造函数将监听类对象传入即可.但是在Activity中不能传递监听对象,所以考虑使用广播来实现. public void login(final LoginOnClickListener listener) { Intent intent = new Intent(context, LoginActivity.clas

C++学习笔记:指向函数的指针

1 #include <stdio.h> 2 3 int sum(int a, int b) 4 { 5 return a+b; 6 } 7 8 int minus(int a, int b) 9 { 10 return a-b; 11 } 12 13 int x(int a, int b) 14 { 15 return a*b; 16 } //第一个参数为指向函数的指针,返回类型为int,参数是int,int 1 void counting(int (*p)(int, int), int a

python学习笔记(03):函数

默认参数值:   只有在行参表末尾的哪些参数可以有默认参数值,即 def func(a, b=5 )#有效的 def func( a=5,b )#无效的 关键参数: #!/usr/bin/python # Filename: func_key.py def func(a, b=5, c=10): print 'a is', a, 'and b is', b, 'and c is', c func(3, 7) func(25, c=24) func(c=50, a=100) #输出: $ pyth

C++学习笔记之字符函数库cctype

C++从C语言继承了一个与字符相关的.非常方便的函数软件包,它可以简化诸如确定字符是否为大写字母.数字.标点符号等工作,这些函数原型是在头文件cctype(老式风格ctype.h)中定义的. 下表对这些函数进行了总结,有些系统可能没有表中列出的函数,也有可能还有在表中没有列出的一些函数. 函数名称 返回值 isalnum() 如果参数是字母数字,即字母或者数字,该函数返回true isalpha() 如果参数是字母,该函数返回true iscntrl() 如果参数是控制字符,该函数返回true

C++ Primer Plus学习笔记之虚函数

C++ Primer Plus学习笔记之虚函数 C++语言的多态性有两种类型:静态多态性和动态多态性.函数重载和运算符重载就是静态多态性的具体表现,而动态多态性是指程序运行过程中才动态的确定操作所针对的对象,它是通过虚函数实现的: 1,虚函数的概念: 一个指向基类的指针可用来指向从基类派生的任何对象,这样就可以达到一个接口多个实现的访问了:虚函数是在基类中被冠以virtual的成员函数,它提供了一种接口界面.虚函数可以在一个或者多个派生类中被重新定义,但要求在派生类中从新定义时,虚函数的函数原型

Oracle 学习笔记 18 -- 存储函数和存储过程(PL/SQL子程序)

PL/SQL子程序 包括函数和过程.这里的函数指的是用户自己定义的函数,和系统函数是不同的.子程序一般是完成特定功能的PL/SQL程序块,并且具有一定的通用性,可以被不同的应用程序多次调用.Oracle提供可以把PL/SQL程序存储在数据库中,并可以再任何地方来运行它.这样就叫做存储过程或者是函数.过程和函数的唯一区别就是函数总是向调用者返回数据,而过程则不返回数据. 函数 如果用户要经常执行某些操作,并且需要返回特定的数据,那么就可以将这些操作构造成一个函数. 可以使用SQL语句定义函数. 基

C++学习笔记32 谓词函数

首先来看看百度百科对"谓词函数"的定义说明: 1定义编辑 一个判断式,一个返回bool值的函数或者仿函数.几元就是函数有几个参数,至于定义和使用,函数定义和一般的函数定义一样,仿函数就是写个类,然后重载operator().使用就是在那些以这种需要返回bool值的函数作参数的函数里用了. 一元谓词函数举例如下 1,判断给出的string对象的长度是否小于6 bool GT6(const string &s) { return s.size() >= 6; } 2,判断给出

C++学习笔记27,虚函数的工作原理

C++规定了虚函数的行为,但是将实现交给了编译器的作者. 通常,编译器处理虚函数的方法是给每一个对象添加一个隐藏成员.隐藏成员中保存了一个指向函数地址数组的指针. 这个数组称为虚函数表(virtual function table,vtbl).虚函数表中存储了为类对象进行声明的虚函数的地址. 例如:基类对象包含一个指针,该指针指向基类的虚函数表. 派生类对象包含一个指针,该指针指向一个独立的虚函数表.如果派生类提供了虚函数的新定义,虚函数表将保存新的函数地址. 如果派生类没有重新定义虚函数,虚函