- struct sockaddr 结构体
- struct sockaddr_in 结构体
- 函数 htons() 、 inet_addr()、inet_aton()
struct sockaddr和struct sockaddr_in这两个结构体用来处理网络通信的地址。
1. sockaddr 定义
sockaddr在头文件include<sys/socket.h中定义,sockaddr的缺陷是:sa_data把目标地址和端口信息混在一起了,如下:
struct sockaddr {
sa_family_t sin_family;//地址族
char sa_data[14]; //14字节,包含套接字中的目标地址和端口信息
};
2. sockaddr_in 定义
sockaddr_in在头文件 include<netinet/in.h 或 include <arpa/inet.h 中定义,该结构体解决了sockaddr的缺陷,把port和addr 分开储存在两个变量中,如下:
struct sockaddr_in {
short int sin_family; // AF_INET,代表了IPV4
unsigned short int sin_port;
struct in_addr sin_addr;
unsigned char sin_zero[8];
}
struct in_addr {
unsigned long s_addr;
}
sin_port和sin_addr都必须是网络字节序(NBO),一般可视化的数字都是主机字节序(HBO)。
3. 总结
sockaddr 常用于bind、connect、recvfrom、sendto等函数的参数,指明地址信息,是一种通用的套接字地址。
sockaddr_in 是internet环境下套接字的地址形式。所以在网络编程中我们会对sockaddr_in结构体进行操作,使用sockaddr_in来建立所需的信息,最后使用类型转化就可以了。
一般先把sockaddr_in变量赋值后,强制类型转换后传入用sockaddr做参数的函数:sockaddr_in用于socket定义和赋值;sockaddr用于函数参数。
二者长度一样,都是16个字节,即占用的内存大小是一致的,因此可以互相转化。二者是并列结构,指向sockaddr_in结构的指针也可以指向sockaddr。
#include <stdio.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define SERV_PORT 4466
#define LOCAL_ADDR "192.168.18.136"
int main(int argc,char **argv)
{
int sockfd;
struct sockaddr_in mysock;
sockfd = socket(AF_INET,SOCK_STREAM,0); //获得fd
bzero(&mysock,sizeof(mysock)); //初始化结构体
mysock.sin_family = AF_INET; //设置地址家族
mysock.sin_port = htons(SERV_PORT); //设置端口
mysock.sin_addr.s_addr = inet_addr(LOCAL_ADDR); //设置地址方法1
//inet_aton(LOCAL_ADDR,&(mysock.sin_addr)); //设置地址方法2
//mysock.sin_addr.s_addr = htons(INADDR_ANY); //设置地址方法3,服务
器端建议用方法3
/*bind的时候进行强制转化*/
bind(sockfd,(struct sockaddr *)&mysock,sizeof(struct sockaddr);
... ...
return 0;
}
上面提到:sin_port和sin_addr都必须是网络字节序(NBO)
题外话,函数 htonl() 和 inet_addr(),inet_aton()。参考这里
-
uint32_t htonl(uint32_t hostlong);
是将主机字节序整数值转换为网络字节序;
头文件: include <arpa/inet.h>。htons() : 由主机字节序转换为网络字节序的16位整数值。(host to net)
ntohs() : 由网络字节序转换为主机字节序的16位整数值。
htonl () : 由主机字节序转换为网络字节序的32位整数值。
ntohl () : 由网络字节序转换为主机字节序的32位整数值。 -
inet_addr(const char *string) :
是将一个IP字符串转化为一个网络字节序的转换函数;
用于sock.sin_addr.s_addr;
头文件:<arpa/inet.h>。mysock.sin_addr.s_addr = inet_addr(local_addr);
-
inet_aton (const char *string, struct in_addr *addr) :
是一个改进的方法来将一个字符串IP地址转换为一个32位的网络序列IP地址;
用于sock.sin_addr;
头文件:<arpa/inet.h>。inet_aton(local_addr,&(mysock.sin_addr));