一、大小端存储概念
数据在各种硬件系统上的排列方式并不是完全统一的,主要分为大端存储和小端存储两种方式。
因为地址是有个顺序的,多个字节的排列也是有个顺序的。
如果是低地址存储了数据的低位,即小端存储模式。
如果是低地址存储了数据的高位,则为大端存储模式。
举个例子,假如这里有个4字节数据:12,34,56,78
(对应的数为305419896,
对应的二进制为:0001 0010, 0011 0100, 0101 0110, 0111 1000)。
如果在内存中的存储方式为:
即为大端存储(数据低字节位78在内存高地址处),
反之,存储方式若为:
则为小端存储(数据低字节位78在内存低地址)。
二、验证本机大小端存储方式
熟悉了大小端存储的定义之后,就可以根据定义来验证本机的大小端存储方式了。
一般会使用c/c++验证,可以使用指针自由操作。
#include <stdio.h>
int main()
{
int a = 1;
char* p = (char*)&(a);
if (*p == 1) {
printf("小端存储");
} else {
printf("大端存储");
}
return 0;
}
为什么这段程序可以验证机器的大小端存储模式呢?
首先,定义了一个int型变量a,一般而言,c的int变量占4个字节,存储的值为1。
那么要验证机器是大端存储还是小端存储,就需要确定1是存储在低字节还是高字节。
这里不能简单的使用如(0x00 & a==0)这样的与运算,或者移位判断,显然这些运算都会根据内存存储模式进行过适配。
这里采用一种截断数据的方法,这里定义了一个char*的变量,虽然指针一般也占4个字节,但char*的指针只存储1个字节的数据。
但是int变量占了4个字节,显然char*存不下,这时char*就会保留低地址里存的数据,放弃高地址的溢出数据。
这样就拿到了低地址的数据了,而低字节的数据我们也有(即1),
所以这里仅需要判断两者是否相等即可,如果相等,则说明低地址存储了低字节数据,否则说明低地址存储了高字节数据。
即可得出本机的内存大小端存储模式。
Python判断本机大小端:
import sys
print(sys.byteorder)
三、socket发送字节流数据的存储模式
socket程序一般发送和接收的数据都是字节流,不同设备的大小端存储模式不一致,就容易导致数据发送和接收出现问题,因此在socket程序中也需要进行一些协议规定。
wwdx.