【TCP/IP】IP地址与域名之间的转换 - gethostbyname 和 gethostbyaddr函数

本文主要是介绍【TCP/IP】IP地址与域名之间的转换 - gethostbyname 和 gethostbyaddr函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

域名系统

DNS服务器

IP地址和域名之间的转换

通过域名获取IP地址

通过IP地址获取域名


域名系统

        域名系统(英文:Domain Name System,缩写:DNS)是互联网的一项服务。它作为将域名和IP地址相互映射的一个分布式数据库,能够使人更方便地访问互联网。

DNS服务器

        所有计算机中会保存记录默认DNS服务器的地址,通过默认DNS服务器便可得到相应域名的IP地址信息。在浏览器地址栏中输入域名后,浏览器会通过默认DNS服务器来获取该域名对应的IP地址信息,从而建立起与目标服务器之间的连接。

补充:

        在Linux系统中,有两条指令可以帮助我们查看DNS服务器地址及其对应的IP地址。

  • ping:能够查看某一域名对应的IP地址
  • nslookup:用于查询DNS的记录,查询域名解析是否正常,在网络故障时用来诊断网络问题

        计算机内置的默认DNS服务器并不知道网络上所有域名的IP地址。当默认DNS服务器无法解析时,则会询问其他DNS服务器,进行多次迭代直到查找到为止,最终把查询记录交还给用户。

        如上图,描述了在默认DNS服务器无法查找到目标域名IP地址时的更迭过程。不难看出,DNS是一种层次化管理的分布式数据库系统。

IP地址和域名之间的转换

        在之前,我们编写的程序都是以IP+端口号的形式来构建连接,然而这只限于IP地址较为固定的场景。当我们需要访问互联网中各类网页时,依靠输入IP地址和端口号这种方式将会显得十分笨拙。(试想如果IP地址发生变化,程序是否就得重新编写)那么,什么方法可以解决这个问题呢?

        域名相对于IP地址,变化更少且稳定,那么利用域名来编写程序便可使维护变得更加轻松。在网络编程中,也有一个合适的函数,能够帮助我们通过域名地址来解析出所需要的各类信息。

通过域名获取IP地址

        使用以下函数可以通过传递域名的字符串形式来获取IP地址。

#include <netdb.h>
struct hostent * gethostbyname(const char * hostname);//成功时返回hostent结构体地址,失败时返回NULL指针

        通过gethostbyname函数,我们只需要传递域名的字符串,就可以得到域名对应的IP地址。但需要注意的是,返回的数据类型是一个结构体形式。该结构体定义如下:

/* Description of data base entry for a single host.  */
struct hostent
{char *h_name;			/* Official name of host.  */char **h_aliases;		/* Alias list.  */int h_addrtype;		/* Host address type.  */int h_length;			/* Length of address.  */char **h_addr_list;		/* List of addresses from name server.  */
#ifdef __USE_MISC
# define	h_addr	h_addr_list[0] /* Address, for backward compatibility.*/
#endif
};

        除了IP地址外,该结构体还带有其他信息。以下是对该结构体中各变量意义的描述:

h_name
用以保存官方域名

h_alìases
用以保存多个可能存在的域名

h_addrtype
用以保存在h_addr_list中IP地址对应的地址族信息。若是IPv4,则变量为AF_INET。
h_length
用以保存IP地址的长度 。若是IPv4地址(4个字节存储) ,则该值为4;IPv6时(16个字节存储),则值为16。

h_addr_list
用以保存多个可能存在的IP地址,变量以char *形式保存域名对应的IP地址 。

        结构体的抽象结构如下:

        让我们尝试一下利用gethostbyname(在netdb.h头文件中)函数来编写一个域名转IP的小程序。

demo_byname.cpp

#include <netdb.h>
#include <iostream>
#include <arpa/inet.h>
int main()
{struct hostent *host;std::string domainAddress;std::cout << "Please input a website addreess" << std::endl;std::cin >> domainAddress;host = gethostbyname(domainAddress.c_str());if (!host){std::cout << "Get host error" << std::endl;return -1;}//输出官方域名std::cout << "Official name:" << host->h_name << std::endl;//输出其他可能存在的多个域名for (size_t i = 0; host->h_aliases[i]; i++){std::cout << "Aliases " << i + 1 << ":" << host->h_aliases[i] << std::endl;}//输出地址族信息std::cout << "Address type: " << ((host->h_addrtype == AF_INET) ? "AF_INET" : "AF_INET6") << std::endl;//输出可能存在的多个(域名)对应的IP地址for (size_t i = 0; host->h_addr_list[i]; i++){std::cout << "IP Address " << i + 1 << ":" << inet_ntoa(*(struct in_addr*)host->h_addr_list[i]) << std::endl;}return 0;
}

输出信息:

        需要注意的是,在h_addr_list这个变量中,虽然数据类型是字符串指针数组,但是元素的实际指向是in_addr结构体变量地址值而非字符串,如图所示。

        因此,在输出时,请根据需要将数据类型作相应的转换。

inet_ntoa(*(struct in_addr*)host->h_addr_list[i]);

        Q:为什么hostent结构体中的成员h_addr_list指向的数组类型是char *,但其形式却是in_addr结构体?

        A:在C支持void *指针之前,在结构定义上,人们把char * 视作“通用指针”。而设计者在设计之初,并未只考虑IPv4,也考虑到了IPv6的格式。在创建结构体时,如果h_addrtype不是AF_INET(IPv4格式),那么h_addr_list中将包含非struct in_addr的结构体类型,设计者为了使结构体具有通用性,用char * 来对这两种不同的结构形式做通用化,这样便免去了重新去设计新的结构体的繁琐过程。(事实也表明,设计者当时做对了,现在struct in_addr将在不久逐渐被IPv6地址所取代)

通过IP地址获取域名

        除了通过域名获取对应的IP地址外,我们也可以通过IP地址来获取域名。gethostbyaddr这个函数可以帮助我们通过传入含有IP地址信息的char *指针(为什么用char *以及涉及到的转换相信也不难理解了),来获取对应的hostent结构体。

#include <netdb.h>struct hostent * gethostbyaddr(const char * addr , socklen_t len , int family);//成功时返回hostent结构体变量地址值,失败时返回NULL指针。/* 变量含义 */
//addr:含有IP地址信息的in_addr结构体指针。为了同时传递IPv4地址之外的其他信息,该变量的类型声明为char指针
//len:向第一个参数传递的地址信息的字节数,IPv4时为4,IPv6时为16
//family:传递地址族信息,IPv4时为AF_INET,IPv6时为AF_INET6

        编写一个demo来验证这个功能吧~

demo_byadd.cpp

#include <netdb.h>
#include <iostream>
#include <arpa/inet.h>
int main()
{struct hostent *host;std::string ipAddress;struct sockaddr_in addr;std::cout << "Please input a website IP addreess:" << std::endl;std::cin >> ipAddress;addr.sin_addr.s_addr=inet_addr(ipAddress.c_str());//注意是在 .sin_addr 处转换为char *,以及注意上面是在 .s_addr 上赋值host = gethostbyaddr((char*)&addr.sin_addr,4,AF_INET);if (!host){std::cout << "Get host error" << std::endl;return -1;}std::cout << "Official name:" << host->h_name << std::endl;for (size_t i = 0; host->h_aliases[i]; i++){std::cout << "Aliases " << i + 1 << ":" << host->h_aliases[i] << std::endl;}std::cout << "Address type: " << ((host->h_addrtype == AF_INET) ? "AF_INET" : "AF_INET6") << std::endl;for (size_t i = 0; host->h_addr_list[i]; i++){std::cout << "IP Address " << i + 1 << ":" << inet_ntoa(*(struct in_addr*)host->h_addr_list[i]) << std::endl;}return 0;
}

输出信息(通过ping 获取到的Google IP地址):

这篇关于【TCP/IP】IP地址与域名之间的转换 - gethostbyname 和 gethostbyaddr函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/165499

相关文章

PostgreSQL中rank()窗口函数实用指南与示例

《PostgreSQL中rank()窗口函数实用指南与示例》在数据分析和数据库管理中,经常需要对数据进行排名操作,PostgreSQL提供了强大的窗口函数rank(),可以方便地对结果集中的行进行排名... 目录一、rank()函数简介二、基础示例:部门内员工薪资排名示例数据排名查询三、高级应用示例1. 每

全面掌握 SQL 中的 DATEDIFF函数及用法最佳实践

《全面掌握SQL中的DATEDIFF函数及用法最佳实践》本文解析DATEDIFF在不同数据库中的差异,强调其边界计算原理,探讨应用场景及陷阱,推荐根据需求选择TIMESTAMPDIFF或inte... 目录1. 核心概念:DATEDIFF 究竟在计算什么?2. 主流数据库中的 DATEDIFF 实现2.1

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串

Javaee多线程之进程和线程之间的区别和联系(最新整理)

《Javaee多线程之进程和线程之间的区别和联系(最新整理)》进程是资源分配单位,线程是调度执行单位,共享资源更高效,创建线程五种方式:继承Thread、Runnable接口、匿名类、lambda,r... 目录进程和线程进程线程进程和线程的区别创建线程的五种写法继承Thread,重写run实现Runnab

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

C# 比较两个list 之间元素差异的常用方法

《C#比较两个list之间元素差异的常用方法》:本文主要介绍C#比较两个list之间元素差异,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. 使用Except方法2. 使用Except的逆操作3. 使用LINQ的Join,GroupJoin

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1