VS2010 C 学习(4):WinSock域名查询解析程序

VS2010 C 学习(4):WinSock 域名查询解析学习VC 编制的Socket 域名查询、解析程序,主要练习网络Winsock 的应用。一、 主要内容:1. 根据IP 地址查询主机信息;

VS2010 C 学习(4):WinSock 域名查询解析

学习VC 编制的Socket 域名查询、解析程序,主要练习网络Winsock 的应用。

一、 主要内容:

1. 根据IP 地址查询主机信息;

2. 根据网址查询主机信息、DNS 解析地址;

,

二、 设计实现:

1. 根据IP 地址查询主机信息;

根据IP 地址调用gethostbyaddr 函数 ,分析hostent 结构体 获取主机信息。

2. 根据网址查询主机信息、DNS 解析地址;

根据网址 调用getaddinfo 函数查询DNS 服务器 ,查询链表依次获取该网址的主机信息,分析SOCKADDR_IN结构体,解析出IP 地址。

三、 基础知识:

(一) 域名解析DNSR(domain name system resolution)

1. 域名解析

域名解析是把域名指向网站空间IP ,让人们通过注册的域名可以方便地访问到网站一种服务。域名解析也叫域名指向、服务器设置、域名配置以及反向IP 登记等等。说得简单点就是将好记的域名解析成IP ,服务由DNS 服务器完成,是把域名解析到一个IP 地址,然后在此IP 地址的主机上将一个子目录与域名绑定。

IP 地址是网路上标识您站点的数字地址,为了方便记忆,采用域名来代替IP 地址标识站点地址。域名解析就是域名到IP 地址的转换过程。域名的解析工作由DNS 服务器完成。

我们知道域名是为了方便记忆而专门建立的一套地址转换系统,要访问一台互联网上的服务器,最终还必须通过IP 地址来实现,域名解析就是将域名重新转换为IP 地址的过程。一个域名对应一个IP 地址,一个IP 地址可以对应多个域名;所以多个域名可以同时被解析到一个IP 地址。域名解析需要由专门的域名解析服务器(DNS ) 来完成。

解析过程,比如,一个域名为:***.com,是想看到这个现HTTP 服务,如果要访问网站,就要进行解析,首先在域名注册商那里通过专门的DNS 服务器解析到一个WEB 服务器的一个固定IP 上:211.214.1.***,然后,通过WEB 服务器来接收这个域名,把***.com这个域名映射到这台服务器上。那么,输入***.com这个域名就可以实现访问网站内容了. 即实现了域名解析的全过程;

人们习惯记忆域名,但机器间互相只认IP 地址,域名与IP 地址之间是对应的,它们之间的转换工作称为域名解析,域名解析需要由专门的域名解析服务器来完成,整个过程是自动进行的。

,

域名解析协议(DNS )用来把便于人们记忆的主机域名和电子邮件地址映射为计算机易于识别的IP 地址。DNS 是一种c/s的结构,客户机就是用户用于查找一个名字对应的地址,而服务器通常用于为别人提供查询服务。

2. TTL 值

全称是“生存时间(Time To Live)”,简单的说它表示DNS 记录在DNS 服务器上缓存时间。

3. A 记录

WEB 服务器的IP 指向A (Address) 记录是用来指定主机名(或域名)对应的IP 地址记录。

(二) Socket

1. socket 定义

socket 接口是TCP/IP网络的API ,socket 接口定义了许多函数或例程,程序员可以用它们来开发TCP/IP网络上的应用程序。

每一个socket 都用一个半相关描述{协议、本地地址、本地端口}来表示;一个完整的套接字则用一个相关描述{协议、本地地址、本地端口、远 程地址、远程端口}来表示。

socket 也有一个类似于打开文件的函数调用,该函数返回一个整型的socket 描述符,随后的连接建立、数据传输等操作都 是通过socket 来实现的。

2. Socket 类型

● 流式socket (SOCK_STREAM)

流式套接字提供可靠的、面向连接的通信流;它使用TCP 协议,从而保证了数据传输的正确性和顺序性。

● 数据报socket (SOCK_DGRAM)

数据报套接字定义了一种无连接的服务,数据通过相互独立的报文进行传输,是无序的,并且不保证是可靠、无差错的。它使用数据报协议UDP 。

● 原始socket

原始套接字允许对底层协议如IP 或ICMP 进行直接访问,它功能强大但使用较为不

,

便,主要用于一些协议的开发。

3. 地址数据结构

● 使用C/C 开发socket 程序时,使用sockaddr 和sockaddr_in这两个结构类型来保存socket 信息。

● 这两个数据类型是等效的,可以相互转化,通常sockaddr_in数据类型使用更为方便。

struct sockaddr

{

unsigned short sa_family; /*协议族*/

char sa_data[14]; /*14字节的 协议地址,包含该socket 的IP 地址和端口号。*/ };

struct sockaddr_in

{

short int sa_family; /*协议族*/

unsigned short int sin_port; /*端口号*/

struct in_addr sin_addr; /*IP地址*/

unsigned char sin_zero[8]; /*填充0 以保持与struct sockaddr同样大小*/ };

4. 协议族

● 上述结构中的sa_family字段用于描述socket 中的协议族,其定义于netinet/in.h

5. 数据存储优先顺序

● 计算机数据存储有两种字节优先顺序:高位字节优先(称为大端模式)和低位字节优先(称为小端模式,PC 机通常采用)。

,

● Internet 上数据以高位字节优先顺序在网络上传输,因此在有些情况下,需要对这两个字节存储优先顺序进行相互转化。

● 对字节存储优先顺序转化可能用到4个函数:htons()、 ntohs()、htonl()和ntohl()。这4个函数分别实现网络字节序和主机字节序的转化,其中h 表示host ,n 表示network ,s 表示short ,l 表示long 。通常16位的IP 端口号用s ,而IP 地址用l 。 ● 调用该函数只是使其得到相应的字节序, 用户不需清楚该系统的主机字节序和网络字节序是否真正相等。如果是相同不需要转换的话,该系统的这些函数会定义成空宏。

6. 字节优先顺序转换函数

7. 地址格式转化

● 通常用户在表达地址时采用的是点分十进制表示的数值(或者是以冒号分开的十进制IPv6地址),而在通常使用的socket 编程中所使用的则是 二进制值,这就需要将这两个数值进行转换。

● 在IPv4中用到的函数有inet_aton()、inet_addr()和inet_ntoa(),而 IPv4和IPv6兼容的函数有inet_pton()和inet_ntop()。由于IPv6是下一代互联网的标准协议,后面涉及的函数都能够同时兼容IPv4和IPv6,但在具体举例时仍以IPv4为例。

● 这里inet_pton()函数是将点分十进制地址映射为二进制地址,而inet_ntop()是

,

将二进制地址映射为点分十进制地址。

8. inet_pton函数

9. inet_ntop函数

10. 主机名

通常,人们在使用过程中都不愿意记忆冗长的IP 地址,尤其到IPv6时,地址长度多达128位。因此,使用主机名将会是很好的选择。

,

● 在Linux 中,同样有一些函数可以实现主机名和地址的转化,最为常见的有gethostbyname()、gethostbyaddr()和getaddrinfo()等,它们都可以实现IPv4和IPv6的地址和主机名之间的转化。其中gethostbyname()将主机名转化为IP 地址,gethostbyaddr()则是逆操作,将IP 地址转化为主机名,另外getaddrinfo()还能实现自动识别IPv4地址和IPv6地址。

● gethostbyname()和gethostbyaddr()都涉及一个hostent 的结构体。

11. hostent 结构体

struct hostent

{

char *h_name; /*正式主机名*/

char **h_aliases; /*主机别名*/

int h_addrtype; /*地址类型*/

int h_length; /*地址字节长度*/

char **h_addr_list; /*指向IPv4或IPv6的地址指 针数组*/

};

● 调用gethostbyname()函数或gethostbyaddr()函数后就能返回hostent 结构体的相关信息。

12. gethostbyname 函数

• 调用该函数时可以首先对hostent 结构体中的h_addrtype和h_length进行设置,若为IPv4可设置为AF_INET和4;若为IPv6可设置为AF_INET6和16;若不设置则默认为IPv4地址类型。

13. getaddrinfo 函数

,

14. addrinfo 结构体

getaddrinfo()函数涉及一个addrinfo 的结构体:

struct addrinfo

{

int ai_flags;

/*AI_PASSIVE, AI_CANONNAME;*/ int ai_family; /*地址族*/

int ai_socktype; /*socket类型*/

int ai_protocol; /*协议类型*/

size_t ai_addrlen; /*地址字节长度*/

char *ai_canonname; /*主机名*/

struct sockaddr *ai_addr; /*socket结构体*/

struct addrinfo *ai_next; /*下一个指针链表*/

};

15. addrinfo 常见选项值

,

• 通常服务器端在调用getaddrinfo()之前,ai_flags设置AI_PASSIVE,用于 bind()函数(用于端口和地址的绑定,后面会讲到),主机名nodename 通常会设置为NULL 。

• 客户端调用getaddrinfo()时,ai_flags一般不设置AI_PASSIVE,但是主机名 nodename 和服务名servname (端口)则应该不为空。

• 即使不设置ai_flags为AI_PASSIVE,取出的地址也可以被绑定,很多程序中ai_flags直接设置为0,即3个标志位都不设置,这种情况 下只要hostname 和servname 设置的没有问题就可以正确绑定。

/*#include files… */

int main()

{

struct addrinfo hints, *res = NULL;

int rc;

memset(&hints, 0, sizeof(hints));

/*设置addrinfo 结构体中各参数 */

hints.ai_flags = AI_CANONNAME;

hints.ai_family = AF_UNSPEC;

hints.ai_socktype = SOCK_DGRAM;

hints.ai_protocol = IPPROTO_UDP;

,

/*调用getaddinfo 函数*/

rc = getaddrinfo("localhost", NULL, &hints, &res);

if (rc != 0)

{

perror("getaddrinfo");

exit(1);

}

else

{

printf("Host name is sn", res->ai_canonname);

}

exit(0);

}

(三) 编程注意

● WS2_32.lib是网络套接字的库。两种方法加入

1、 菜单的 项目 - 属性 - Linker - Input - Additional Dependencies 加上

2、 直接在代码里面 #pragma comment(lib, "ws2_32.lib")

● WSAStartup

使用Winsock 库函数之前, 必须先调用函数WSAStartup, 该函数负责初始化动态连接库Ws2_32.dll.

函数定义:

int WSAStartup ( WORD wVersionRequested, LPWSADATA lpWSAData );

● wVersionRequested

标签: