Address模块
Linux
使用Berkeley套接字接口进行网络编程,目前基本所有的系统上都支持。Berkeley
套接字大部分API
都需要传入一个地址作为参数,用于表示网络中的一台主机的通信地址。但是不同的协议类型对应的网络地址类型不一样。比如IPv4
协议对应IPv4
地址,其长度为32
位,而IPv6
协议对应IPv6
地址,其长度为128
位
如果针对每种类型的地址都制定一套对应的API
,那么套接字的API
数量就会相对宠大,这对开发和维护都没好处。我们希望的是只使用一套通用的API
就能实现各种地址类型的操作,比如针对IPv4
地址的代码,能够在不修改或尽量少修改的前提下,就可用于IPv6
地址。对此,Berkeley
套接字接口拟定了一个通用套接字地址结构sockaddr
,用于表示任意类型的地址,所有的套接字API
在传入地址参数时都只需要传入sockaddr
类型,以保证接口的通用性。除通用地址结构sockaddr
外,还有一系列表示具体的网络地址的结构,这些具体的网络地址结构用于用户赋值,但在使用时,都要转化成sockaddr
的形式
sockaddr
表示通用套接字地址结构,其定义如下:
struct sockaddr {
unsigned short sa_family; // 地址族(地址类型)
char sa_data[14]; // 地址内容
};
sockaddr_in
表示IPv4
地址结构,其定义如下:
struct sockaddr_in {
unsigned short sin_family; // 地址族,IPv4的地址族为AF_INET
unsigned short sin_port; // 端口
struct in_addr sin_addr; // IP地址,IPv4的地址用一个32位整数来表示
char sin_zero[8]; // 填充位,填零即可
};
sockaddr_in6
表示IPv6
地址结构,其定义如下:
struct sockaddr_in6 {
unsigned short sin6_family; // 地址族,IPv6的地址族为AF_INET6
in_port_t sin6_port; // 端口
uint32_t sin6_flowinfo; // IPv6流控信息
struct in6_addr sin6_addr; // IPv6地址,实际为一个128位的结构体
uint32_t sin6_scope_id; // IPv6 scope-id
};
通过上面的定义也可以发现,除sockaddr_in
可以无缝转换为sockaddr
外,sockaddr_in6
和sockaddr_un
都不能转换为sockaddr
,因为大小不一样
但这并不影响套接字接口的通用性,Berkeley套接字接口都是以指针形式接收sockaddr参数,并且额外需要一个地址长度参数,又由于以上所有的地址结构的前两个字节都表示地址族,所以通过sockaddr
指针总能拿到传入地址的地址类型,通过地址类型判断出地址长度后,再通过sockaddr
指针取适合该地址的长度即可拿到地址内容。比如上面的地址内容占14
字节,这并不足以容纳一个128
位16
字节的IPv6
地址。但当以指针形式传入时,完全可以通过指针取到适合IPv6
的长度
github
https://github.com/huxiaohei/tiger.git
实现
tiger
采用Berkeley
同样的设计思路,将IPv4
、IPv6
网络地址和Unix
本地地址,以及与地址先关的基本操作封装在Address
模块中,方便后续对地址的使用和维护