大小端的定义
小端:数据低位存在低地址中,数据的高位存在高地址中
大端:数据低位存在高地址中,数据的高位存在低地址中
指针和大小端的结合使用
如果是用一个指针指向存储的这块内存,不管是大端还是小端,指向的都是这块内存的最地位地址。正是由于指向的是最低位地址,正好可以使用这个特性,来检查机器是大端还是小端。
int _tmain(int argc, _TCHAR* argv[])
{
//int a[2] = {16909060,84281096};
int a[2] = {0x01020304,0x05060708};
char *pa =(char*) &a;
int i = 8;
while(i> 0)
{
printf("%02x ",*pa);
pa++;
i--;
}
printf("\n");
}
输出结果为,证明机器为小端
大小端的读取过程
读数据永远是从低地址开始的
发送数据是从低地址开始发送
收到数据是从低地址开始存储
我们知道这是小端存储,所以在读出来的时候会从低位开始放
我们知道这是大端存储,所以在读出来的时候会从高位开始放
大小端在网络中传输的细节
网络字节序是大端字节序
UDP/TCP/IP协议规定:把接收到的第一个字节当作高位字节看待,这就要求发送端发送的第一个字节是高位字节;而在发送端发送数据时,发送的第一个字节是该数值在内存中的起始地址处对应的那个字节,也就是说,该数值在内存中的起始地址处对应的那个字节就是要发送的第一个高位字节(即:高位字节存放在低地址处);由此可见,多字节数值在发送之前,在内存中因该是以大端法存放的;所以,在实际中传输中,当在两个存储方式不同的主机上传输时,需要借助字节序转换函数
如果小端向大端设备传送 0x 01 02 03 04 05 06 07 08 八个字节,假设是两个int
其在小端存储为
小端发送是从低地址开始读取,又由于发送的第一个字节需要是高位字节,所以需要网络序转换函数htonl将两个int转换成大端序,如下图,发送的顺序为 01 02 03 04 …… 08
大端接收到数据,把收到的第一个自己当做高位,即收到的数据如下图
按照两个字节读取,所以收到的第一个int是 0x01020304 第二个int是 0x05060708,正确
高位和低位的概念
https://blog.csdn.net/error_again/article/details/107457067
大小端
https://zhuanlan.zhihu.com/p/144718837
位域
https://zhuanlan.zhihu.com/p/67725528