内容
第3章 地址族与数据序列
IPv4 - 4字节表示IP地址,即32bit
IPv6 - 6字节表示IP地址,即48bit
一个IPv4地址包括网络地址和计算机地址两部分,根据网络地址占用字节数的不同,可分为:
A类,1个字节的网络地址+3个字节的计算机地址
B类,2个字节的网络地址+2个字节的计算机地址
C类,3个字节的网络地址+1个字节的计算机地址
D类,4个字节的网络地址,是多播IP地址
E类,被预约了的地址
只需通过IP地址的第一个字节即可判断网络地址占用的字节数:
A类地址的首字节范围是:0(0000,0000 ) - 127(0111,1111)
B类地址的首字节范围是:128(1000,0000 ) - 191(1011,1111)
C类地址的首字节范围是:192(1100,0000) - 223(1101,1111)
IP地址用于找到对应的NIC(网卡)传输数据,端口号用于操作系统把数据传递给相应的套接字。网卡(NIC)接收到的网络数据里含有端口号。端口号和套接字一一对应。
struct sockaddr
{
sa_family_t sin_family;
char sa_data[14];
}
^
||
struct sockaddr_in
{
sa_family_t sin_family; //地址族 [AF_INET, AF_INET6, AF_LOCAL]
uint16_t sin_port; //16位端口号, 网络字节序
struct in_addr sin_addr; //32位IP地址,网络字节序
char sin_zero[8]; //不使用
}
struct in_addr
{
in_addr_t s_addr; //in_addr_t = uint32_t = unsigned 32-bit int; 32位IPv4地址
}
函数bind()中第二个输入参数类型sockaddr,因为该结构体第二部分不易理解和生成,故定义了一个新的结构体sockaddr_in。
CPU向内存读存数据(如0x12345678)的方式有两种:
大端序:高位字节存放到低位地址,如 [20号内存]0x12, [21号内存]0x34, [22号内存]0x56, [23号内存]0x78
小端序:高位字节存放到高位地址,如 [20号内存]0x78, [21号内存]0x56, [22号内存]0x34, [23号内存]0x12
Intel/AMD系列CPU采用小端序存取数据。网络数据传输时约定为大端序,即网络字节序是大端序。
字节序转换函数
unsigned short htons(unsigned short); // h(host) to n(net) s(short,2字节,端口号)
unsigned short ntohs(unsigned short); // n(net) to h(host) s(short,2字节,端口号)
unsigned long htonl(unsigned long); //h(host) to n(net) l (long, 4字节,IP地址)
unsigned long ntohl(unsigned long); //n(net) to h(host) l (long, 4字节,IP地址)
字节序转换函数只需在向sockaddr_in结构体变量赋值时调用,其他情况无需考虑。
ip地址格式转换
#include <arpa/inet.h>
in_addr_t inet_addr(const char * string); //从字符串ip(如1.2.3.4)转换成4字节unsigned int(0x04030201,大端序)
int inet_aton(const char * string, struct in_addr * addr); //结构体in_addr是sockaddr_in中用于表示ipv4地址的结构
char * inet_ntoa(struct in_addr adr);