地址族与数据序列
分配给套接字的IP地址与端口号
IP是网络协议的简写,为收发网络数据而分配给计算机的值。端口号是为区分程序中创建的套接字而分配给套接字的序号。
网络地址
两种IP地址:
IPv4和IPv6的主要差别是表示IP地址所用的字节数。
IPv4标准的4字节IP地址分为网络地址和主机地址。且分为ABCD等类型。
网络地址(网络ID)是为区分网络而设置的一部分IP地址。
传输数据时,并非一开始就浏览所以4字节的IP地址,而是先浏览IP地址中的网络地址,先把数据传到网络。构成网络的路由器接到数据后,浏览传输数据的主机地址并将数据传给目标计算机。
所以,向相应网络传输数据实际上是向构成网络的路由器或交换机传递数据,由接受数据的路由器根据数据中的主机地址向目标主机传递数据。
网络地址分类与主机地址边界
只需通过IP地址的第一个字节即可判断网络地址所占用的字节数,根据IP地址的边界区分网络地址:
正因如此,通过套接字收发数据时,数据传到网络后即可轻松找到正确的主机。
用于区分套接字的端口号
IP用于区分计算机,只要有IP地址就能向目标追击传输数据,但仅凭这些无法传输给最终的应用程序。
计算机中一般配有NIC(网络接口卡)数据传输设备。通过NIC向计算机内部传输数据时会用到IP。操作系统负责把传递到内部的数据适当分配给套接字,就要利用端口号。
通过NIC接受的数据内有端口号,操作系统参考端口号把数据传输给相应端口的套接字:
端口号时同一操作系统内为区分不同套接字而设置的。因此无法将1个端口号分配给不同套接字。
虽然端口号不能重复,但TCP套接字和UDP套接字不会共用端口号,所以允许重复。
如:某TCP套接字使用9190号端口,则其他TCP套接字无法使用该端口号,但UDP套接字可以使用。
总之,数据传输目标地址同时包含IP地址和端口号,只有这样,数据才会被传输到最终的目的应用程序。
地址信息的表示
表示IPv4地址的结构体
-
struct sockaddr_in
-
{
-
sa_family_t family_t; //地址族
-
uint16_t sin_port; //16位TCP/UDP端口号
-
struct in_addr sin_addr; //32位IP地址
-
char sin_zero[8]; //不使用
-
}
-
-
struct in_addr
-
{
-
in_addr_t s_addr; //32位的IPv4地址
-
}
//参考POSIX中定义的数据类型
结构体sockaddr_in的成员分析
sockaddr_in的结构体变量地址值将以如下方式传递给bind函数。
网络字节序与地址变换
字节序与网络字节序
CPU数据保存方式分为大端和小端。主流CPU以小端序方式保存数据。
分析字节序不同的计算机之间数据传递过程:
大端序系统传输数据0x1234时,发送顺序为0x12,0x34; (从保存在地位地址的数据开始传输)
接受端以小端序保存数据,因此接受的数据变为0x3412;
正因如此,在通过网络传输数据时约定统一方式,这种约定成为网络字节序,非常简单---全统一为大端序。
即先把数据数组转化成大端序格式再进行网络传输。
字节序转换
-
unsigned short htons(unsigned short); //h to n把short型数据从主机字节序转化为网络字节序
-
unsigned short ntohs(unsigned short); //n to h把short型数据从网络字节序转化为主机字节序
-
unsigned long htonl(unsigned long); //h to n..long....
-
unsigned long ntohl(unsigned long); //n to h..long...
htons中的h代表主机(host)字节序
htons中的n代表网络(network)字节序
s代表short,l代表long
通常,以s作为后缀的函数中,s代表2个字节short,因此用于端口号转换;以l作为后缀的函数中,l代表4个字节,因此用于IP地址转换。
例子:以上函数的调用:
-
//endian.conv.c
-
-
-
-
int main(int argc, char *argv[])
-
{
-
unsigned short host_port = 0x1234;
-
unsigned short net_port;
-
-
unsigned long host_addr = 0x12345678;
-
unsigned long net_addr;
-
-
net_port = htons(host_port);
-
net_addr = htonl(host_addr);
-
-
printf("Host ordered port: %#x ",host_port);
-
printf("Network ordered port: %#x ",net_port);
-
printf("Host ordered address: %#lx ",host_addr);
-
printf("Network ordered address: %#lx ",net_addr);
-
-
return 0;
-
}
在小端序CPU中运行的结果:
网络地址的初始化与分配
将字符串信息转换为网络字节序的整型数
sockaddr_in保存地址信息的成员为32位整型。为了分配IP地址,需要将其表示为32为整数型数据。
IP地址的表示是点分十进制表示法(192.168.1.1),函数inet_addr可以将字符串形式的IP地址转换成32位整形地址。并且同时进行网络字节序转换。
inet_addr函数:
类型in_addr_t在内部声明为32位整数型。
-
/* inet_addr函数将点分十进制转换为32为整型数据。并且该整型值满足网络字节序 */
-
-
-
-
int main(int argc,char *argv[])
-
{
-
char *addr1 = "1.2.3.4";
-
char *addr2 = "1.2.3.256"; //256是错误的IP地址,检测该函数的错误检测能力
-
-
unsigned long conv_addr = inet_addr(addr1);
-
if (conv_addr == INADDR_NONE)
-
printf("Error! ");
-
else
-
printf("Network ordered integer addr: %#lx ",conv_addr);
-
-
conv_addr = inet_addr(addr2);
-
if(conv_addr == INADDR_NONE)
-
printf("Error! ");
-
else
-
printf("Network ordered integer addr: %#lx ",conv_addr);
-
-
return 0;
-
}
inet_aton函数:与inet_addr函数在功能上完全相同,只不过函数利用了in_addr结构体
inet_ntoa函数:将网络字节序整数IP地址转换成熟悉的字符串(点分十进制形式)
网络地址初始化
介绍套接字创建过程中常见的网络地址信息初始化方法
-
struct sockaddr_in addr;
-
char *serv_ip = "211.217.168.13"; //声明IP地址字符串
-
char *serv_port = "9190"; //声明端口号字符串
-
memset(&addr,0,sizeof(addr)); //结构体变量addr的所有成员初始化为0
-
addr.sin_family = AF_INET; //指定地址族
-
addr.sin_addr.s_addr = inet_addr(serv_ip); //基于字符串的IP地址初始化
-
addr.sin_port = htons(atoi(serv_port)); //基于字符串的端口号初始化
memset函数用来初始化addr,每个字节都初始化为同一个值(0);最后一个参数传入addr的长度
客户端地址信息初始化
上述网络地址信息初始化过程主要针对服务器端而非客户端。给套接字分配IP地址和端口号主要为下面的事作准备:
反观客户端中连接请求如下:
服务器端的准备工作通过bind函数完成,而客户端则通过connect函数完成。
服务器端声明sockaddr_in结构体变量,将其初始化为赋予服务器端IP和套接字的端口号,然后调用bind函数;
客户端则声明sockaddr_in结构体,并初始化为要与之连接的服务器端套接字的IP和端口号。然后调研connect函数.
INADDR_ANY
利用常数INADDR_ANY分配服务器端的IP地址。这种方式可以自动获取运行服务器端的计算机IP地址,不必亲自输入。
向套接字分配网络地址
bind将初始化后的地址信息分配给套接字。
服务器端常见套接字初始化过程
-
int serv_sock;
-
struct sockaddr_in serv_addr;
-
char *serv_port = "9190";
-
-
//创建服务器端套接字(监听套接字)
-
serv_sock = socket(PF_INET,SOCK_STREAM,0);
-
-
//地址信息初始化
-
memset(&serv_addr,0,sizeof(serv_addr));
-
serv_addr.sin_family = AF_INET;
-
serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
-
serv_addr.sin_port = htons(atoi(serv_port));
-
-
//分配地址信息
-
bind(serv_sock,(struct sockaddr *)&serv_addr, sizeof(serv_addr));
-
....