首先来看sockaddr,sockaddr是通用地址结构,目的是为了使不同格式的地址能够被传入到套接字函数。但是通用只是规定了名字和两个主要字段:
struct socketaddr
{
sa_family_t sa_family;
char sa_data[];
...
};
至于sa_data的长度和剩余的扩展字段都是由各实现自己定义。在linux中是这么定义的:
struct socketaddr
{
sa_family_t sa_family;
char sa_data[14];
};
其次来看sockaddr_in,在AF_INET ipv4因特网域中,规定了通用的sockaddr_in结构体:
struct sockaddr_in
{
sa_family_t sin_family;
in_port_t sin_port;
struct in_addr sin_addr;
...
};
在此结构在linux的定义是如下的:
struct sockaddr_in {
sa_family_t sin_family;
in_port_t sin_port;
struct in_addr sin_addr;
unsigned char sin_zero[8];
};
其中sin_zero为填充字段,必须全部置为0;而结构体in_addr的定义为:
struct in_addr
{
in_addr_t s_addr;
};
in_addr_t定义为uint32_t.
in_port_t定义为uint16_t.
现在再来计算一下socketaddr和socketaddr_in的长度:
socketaddr_in的sin_family以外的字段的长度:sin_port(2) + sin_addr(4) + sin_zero(8) = 14;而巧合的是socketaddr的sa_data的长度也为14
由此我们猜想这两个字段大小和内容应该是一样的,实际上确实如此,在linux的ipv4域中我们可以安全地把指向其中一个结构的指针强制转型为另一个。