人防网站建设与服务,网络架构和现实架构的差异,做外贸找客户的网站,wordpress简历模板struct sockaddr和struct sockaddr_in这两个结构体用来处理网络通信的地址。 一、sockaddr sockaddr在头文件#include sys/socket.h中定义#xff0c;sockaddr的缺陷是#xff1a;sa_data把目标地址和端口信息混在一起了#xff0c;如下#xff1a;
struct sockad…struct sockaddr和struct sockaddr_in这两个结构体用来处理网络通信的地址。 一、sockaddr sockaddr在头文件#include sys/socket.h中定义sockaddr的缺陷是sa_data把目标地址和端口信息混在一起了如下
struct sockaddr { sa_family_t sin_family;//地址族 char sa_data[14]; //14字节包含套接字中的目标地址和端口信息 }; 二、sockaddr_in sockaddr_in在头文件#includenetinet/in.h或#include arpa/inet.h中定义该结构体解决了sockaddr的缺陷把port和addr 分开储存在两个变量中如下 struct sockaddr_in
{/*地址族一般来说AF_INET地址族PF_INET协议族*/short sin_family;/*端口号必须要采用网络数据格式,普通数字可以用htons()函数转换成网络数据格式的数字)*/unsigned short sin_port;/*IP地址 in network byte orderInternet address*/struct in_addr sin_addr;/*Same size as struct sockaddr没有实际意义,只是为了跟SOCKADDR结构在内存中对齐*/unsigned char sin_zero[8];
};
sin_port和sin_addr都必须是网络字节序NBO一般可视化的数字都是主机字节序HBO。
其中结构体in_addr 用来表示一个32位的IPv4地址. in_addr_t 一般为 32位的unsigned int其字节顺序为网络顺序network byte ordered)即该无符号整数采用大端字节序 。.其中每8位代表一个IP地址位中的一个数值. 例如192.168.3.144记为0x9003a8c0,其中 c0 为192 ,a8 为 168, 03 为 3 , 90 为 144 打印的时候可以调用inet_ntoa()函数将其转换为char *类型.
struct in_addr { in_addr_t s_addr; };
使用例子 struct sockaddr_in servaddr, cliaddr;bzero(servaddr, sizeof(servaddr));servaddr.sin_family AF_INET;servaddr.sin_addr.s_addr htonl(INADDR_ANY);//将一个32位数从主机字节顺序转换成网络字节顺序servaddr.sin_port htons(PORT); //#define PORT 8889 三、总结 二者长度一样都是16个字节即占用的内存大小是一致的因此可以互相转化。二者是并列结构指向sockaddr_in结构的指针也可以指向sockaddr。
sockaddr常用于bind、connect、recvfrom、sendto等函数的参数指明地址信息是一种通用的套接字地址。 sockaddr_in 是internet环境下套接字的地址形式。所以在网络编程中我们会对sockaddr_in结构体进行操作使用sockaddr_in来建立所需的信息最后使用类型转化就可以了。一般先把sockaddr_in变量赋值后强制类型转换后传入用sockaddr做参数的函数sockaddr_in用于socket定义和赋值sockaddr用于函数参数。
例子1如下
#include stdio.h
#include stdlib.h
#include sys/socket.h
#include netinet/in.hint main(int argc,char **argv)
{int sockfd;struct sockaddr_in mysock;sockfd socket(AF_INET,SOCK_STREAM,0); //获得fdbzero(mysock,sizeof(mysock)); //初始化结构体mysock.sin_family AF_INET; //设置地址家族mysock.sin_port htons(800); //设置端口mysock.sin_addr.s_addr inet_addr(192.168.1.0); //设置地址bind(sockfd,(struct sockaddr *)mysock,sizeof(struct sockaddr); /* bind的时候进行转化 */... ...return 0;
}
例子2如下 int socket_fd ;unsigned short localPort 8090unsigned int localPort “192.168.0.0”;socket_fd socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);struct sockaddr_in local_addr;local_addr.sin_family AF_INET;local_addr.sin_port htons(localPort);local_addr.sin_addr.s_addr htonl(localIP);rtnValue bind(socket_fd, reinterpret_caststruct sockaddr*(local_addr), sizeof(struct sockaddr));if(0 rtnValue){printf(OpenPort LocalIP %s LocalPort%d socket%d.\r\n, inet_ntoa(local_addr.sin_addr), localPort,socket_fd);}else{printf(OpenPort failed!! LocalIP %s LocalPort%d socket%d.\r\n, inet_ntoa(local_addr.sin_addr),localPort,socket_fd);}
两个函数 htons() 和 inet_addr()。
端口号转换函数htons()作用是将端口号由主机字节序转换为网络字节序的整数值。(host to net)
IP地址转换函数inet_addr()作用是将一个IP字符串转化为一个网络字节序的整数值用于sockaddr_in.sin_addr.s_addr。inet_ntoa()作用是将一个sin_addr结构体输出成IP字符串(network to ascii)。比如 printf(%s,inet_ntoa(mysock.sin_addr)); htonl()作用和htons()一样不过它针对的是32位的long而htons()针对的是两个字节16位的short。
与htonl()和htons()作用相反的两个函数是ntohl()和ntohs()。 参考 《TCP/IP网络编程》