分歧CPU中,4字节整数1在内存空间的存储方法是分歧的。4字节整数1可用2进制表现如下:
00000000 00000000 00000000 00000001
有些CPU以下面的次序存储到内存,别的一些CPU则以倒序存储,如下所示:
00000001 00000000 00000000 00000000
若不思索这些就收发数据会发作成绩,由于保管次序的分歧意味着对接纳数据的解析次序也分歧。
CPU向内存保管数据的方法有两种:
大端序(Big Endian):高位字节寄存到低位地址(高位字节在前)。
小端序(Little Endian):高位字节寄存到高位地址(低位字节在前)。
仅凭描绘很难说明清晰,无妨来看一个实例。假定在 0x20 号开端的地址中保管4字节 int 型数据 0x12345678,大端序CPU保管方法如下图所示:
图1:整数 0x12345678 的大端序字节表现
关于大端序,最高位字节 0x12 寄存到低位地址,最低位字节 0x78 寄存到高位地址。小端序的保管方法如下图所示:
图2:整数 0x12345678 的小端序字节表现
分歧CPU保管息争析数据的方法分歧(主流的Intel系列CPU为小端序),小端序零碎和大端序零碎通讯时会发作数据解析毛病。因而在发送数据前,要将数据转换为一致的格局——收集字节序(Network Byte Order)。收集字节序一致为大端序。
主机A先把数据转换成大端序再停止收集传输,主机B收到数据后先转换为本人的格局再解析。
在《运用bind()和connect()函数》一节中解说了 sockaddr_in 构造体,个中就用到了收集字节序转换函数,如下所示:
//创立sockaddr_in构造体变量 struct sockaddr_in serv_addr; memset(&serv_addr, 0, sizeof(serv_addr)); //每一个字节都用0填充 serv_addr.sin_family = AF_INET; //运用IPv4地址 serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1"); //详细的IP地址 serv_addr.sin_port = htons(1234); //端标语
htons() 用来将以后主机字节序转换为收集字节序,个中h代表主机(host)字节序,n代表收集(network)字节序,s代表short,htons 是 h、to、n、s 的组合,可以了解为”将short型数据从以后主机字节序转换为收集字节序“。
罕见的收集字节转换函数有:
htons():host to network short,将short类型数据从主机字节序转换为收集字节序。
ntohs():network to host short,将short类型数据从收集字节序转换为主机字节序。
htonl():host to network long,将long类型数据从主机字节序转换为收集字节序。
ntohl():network to host long,将long类型数据从收集字节序转换为主机字节序。
平日,以s为后缀的函数中,s代表2个字节short,因而用于端标语转换;以l为后缀的函数中,l代表4个字节的long,因而用于IP地址转换。
举例阐明上述函数的挪用进程:
#include <stdio.h> #include <stdlib.h> #include <WinSock2.h> #pragma comment(lib, "ws2_32.lib") int main(){ unsigned short host_port = 0x1234, net_port; unsigned long host_addr = 0x12345678, net_addr; net_port = htons(host_port); net_addr = htonl(host_addr); printf("Host ordered port: %#x\n", host_port); printf("Network ordered port: %#x\n", net_port); printf("Host ordered address: %#lx\n", host_addr); printf("Network ordered address: %#lx\n", net_addr); system("pause"); return 0; }
运转后果:
Host ordered port: 0x1234
Network ordered port: 0x3412
Host ordered address: 0x12345678
Network ordered address: 0x78563412
别的需求阐明的是,sockaddr_in 中保管IP地址的成员为32位整数,而我们熟习的是点分十进制表现法,例如 127.0.0.1,它是一个字符串,因而为了分派IP地址,需求将字符串转换为4字节整数。
inet_addr() 函数可以完成这种转换。inet_addr() 除了将字符串转换为32位整数,同时还停止收集字节序转换。请看下面的代码:
#include <stdio.h> #include <stdlib.h> #include <WinSock2.h> #pragma comment(lib, "ws2_32.lib") int main(){ char *addr1 = "1.2.3.4"; char *addr2 = "1.2.3.256"; unsigned long conv_addr = inet_addr(addr1); if(conv_addr == INADDR_NONE){ puts("Error occured!"); }else{ printf("Network ordered integer addr: %#lx\n", conv_addr); } conv_addr = inet_addr(addr2); if(conv_addr == INADDR_NONE){ puts("Error occured!"); }else{ printf("Network ordered integer addr: %#lx\n", conv_addr); } system("pause"); return 0; }
运转后果:
Network ordered integer addr: 0x4030201
Error occured!
从运转后果可以看出,inet_addr() 不只可以把IP地址转换为32位整数,还可以检测有效IP地址。
留意:为 sockaddr_in 成员赋值时需求显式地将主机字节序转换为收集字节序,而经过 write()/send() 发送数据时TCP协定会主动转换为收集字节序,不需求再挪用响应的函数。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。