IPv4地址结构体sockaddr_in详解
sockaddr_in结构体定义
struct sockaddr_in { sa_family_t sin_family; //地址族(Address Family) uint16_t sin_port; //16位TCP/UDP端口号 struct in_addr sin_add; //32位IP地址 char sin_zero[8]; //不使用 }; struct in_addr { In_addr_t s_addr; //32位IPv4地址 };
成员sin_family
地址族(Address Family) | 含义 |
---|---|
AF_INET | IPv4网络协议中使用的地址族 |
AF_INET6 | IPv6网络协议中使用的地址族 |
AF_LOCAL | 本地通信中采用的Unix协议的地址族 |
成员sin_port
该成员保存16位端口号,重点在于,他以网络字节序保存。
成员sin_addr
该成员保存32位IP地址信息,且也以网络字节序保存。
成员sin_zero
无特殊含义。只是为了使结构体sockaddr_in的大小与sockaddr结构体保持一致而插入的成员。必须填充为0,否则无法得到想要的结果。
下面给出sockaddr_in结构体传递给bind()函数的代码
struct sockaddr_in serv_addr .... if( bind(serv_sock, (struct sockaddr *) &serv_addr), sizeof(serv_addr) == -1 ) //错误处理 ....
请注意第二个参数将sockaddr_in类型转换为sockaddr类型。因为bind()函数需要一个sockaddr类型的参数,但是直接向sockaddr结构体填充地址族、端口号、IP地址等信息不太容易,而填充sockaddr_in类型结构体则容易得多,故先填充sockaddr_in再将其转换为sockaddr类型。
下面给出sockaddr结构体定义。
struct sockaddr { sa_family_t sin_family; //地址族(Address Family) char sa_data[14]; //地址信息 }
此结构体成员sa_data保存的地址信息中需包含IP地址和端口号,剩余部分应填充0,这也是bind()函数要求的。而这对于包含地址信息来讲非常麻烦,继而就有了结构体sockaddr_in。若按照之前讲解的填写好sockaddr_in结构体,则将生产负荷bind()函数要求的字节流。最后转换为sockaddr类型的结构体变量,再传递给bind()函数即可。
会不会有这样的疑问,sockaddr_in是保存IPv4地址信息的结构体,那为何还需要通过sin_family单独指定地址族信息呢?这与sockaddr结构体有关。结构体sockaddr并非只为IPv4设计的。这从保存地址信息的数组sa_data长度为14字节也可看出。因此,结构体sockaddr要求在sin_family中指定地址族信息。为了与sockaddr保持一致,sockaddr_in结构体中也有地址族信息。
上一篇: python移位运算符