在指针(1)这篇文章里,提到一个大小端的概念,这里就介绍一下大小端。
大小端是一种数据的存储方式,分为大端字节序和小端字节序。
大端字节序:数值小的数据保存在高地址,数值大的数据保存在低地址
小端字节序:数值小的数据保存在低地址,数值大的数据保存在高地址
看下面这段代码
int main()
{
int a = 0x11223344;
char* ch = (char*)&a;
*ch = 0;
printf("%x", a);
}
a里保存了一个16进制的数字11223344,将a的地址取出,强转为char*类型,保存在ch这个指针变量,对ch进行解引用并赋值为0,最后以16进制的形式打印a的值,这段代码就解读完了。
运行结果如下
可以看到,最后的44改为了00,通过指针(1)的学习,知道了,char*类型的指针解引用可以访问一个字节。
打开内存窗口,可以清晰的看到每个字节存储的内容
从上往下,地址依次增加,在第一个地址处原本存储的44被改为了00,后面依次为33,22,11。对于0x11223300这个数来说,11是数值较大的一端,也就是通常说的高位,00这一端是低位,数值大的存在了高地址处,数值小的存在了低地址处,所以我的电脑是以小端字节序存储。
每个机器的存储方式不一样,如何写一个代码检查机器是大端存储还是小端存储?
这里给出两种方法。
法一
int main()
{
int i = 1;
char* ch = (char*)&i;
if (*ch == 1)
{
printf("小端存储");
}
else
{
printf("大端存储");
}
}
十进制的1转化为16进制的1为0x00000001,如果*ch==1,那么代表数据小的存储在低地址处。
内存图如下
法二 利用联合体的特点:共用同一块空间。
代码如下
union un
{
int i;
char c;
};
int main()
{
union un u = { 1 };
printf("%d", u.c);
}
创建一个联合体变量u,初始化成员i的值为1,如果是小端存储,则会打印1,反之会打印0。
大小端的介绍就到这里了,大家可以去测试一下自己的机器是大端存储还是小端存储。