大小端概念
大端字节序存储:数据的高位字节保存在内存的低地址中,而数据的低位字节保存在内存的高地址中。
小端字节序存储:数据的高位字节保存在内存的高地址中,而数据的低位字节保存在内存的低地址中。
为什么要有大端和小端之分?
这是因为在计算机系统中,我们是以字节为单位的,每个地址单元都对应着一个字节,一个字节为8bit。但是在C语言中除了8bit的char之外,还有16bit的short型,32bit的long型(要看具体的编译器),另外,对于位数大于8位的处理器,例如16位或者32位的处理器,由于寄存器宽度大于一个字节,那么必然存在着一个如果将多个字节安排的问题。因此就导致了大端存储模式和小端存储模式。
例如:一个 16bit 的 short 型 x ,在内存中的地址为 0x0010 , x 的值为 0x1122 ,那么 0x11 为高字节, 0x22为低字节。对于大端模式,就将 0x11 放在低地址中,即 0x0010 中, 0x22 放在高地址中,即 0x0011 中。小端模式,刚好相反。有些ARM处理器还可以由硬件来选择是大端模式还是小端模式。
请简述大端字节序和小端字节序的概念,设计一个小程序来判断当前机器的字节序(百度笔试题)
思路:假设给上int a = 1;1在内存中16进制表示形式为0x 00 00 00 01,那么这段数据如果按大小端放置:
大端:00 00 00 01
小端:01 00 00 00
可以看出,可以通过判断第一个字节的内容判断大小端,第一个字节为0则是大端,为1则是小端。
可是怎么取第一个字节呢?这里我们可以用指针的类型的解引用来控制访问内存的大小,char*就是只访问一个字节的内存。
此时可以用一个函数来解决
int check_sys()
{
int a = 1;//0x 00 00 00 01
char* p = (char*)&a;//用int访问的是00 00 00 01;强制类型转换为char后访问的是一个字节。
return *p;
}
若返回值是 1 则说明是小端存储
若返回值是 0 则说明是大端存储
完整代码如下:
#include <stdio.h>
//判断机器序的大小端
int check_sys()
{
int a = 1;//0x 00 00 00 01
char* p = (char*)&a;//用int访问的是00 00 00 01;强制类型转换为char后访问的是一个字节。
return *p;
}
int main()
{
int ret = check_sys();
if (ret == 1)
{
printf("小端\n");
}
else
{
printf("大端\n");
}
return 0;
}
此代码可以优化:
#include <stdio.h>
//判断机器序的大小端
int check_sys()
{
int a = 1;//0x 00 00 00 01
return * (char*)&a;//用int访问的是00 00 00 01;强制类型转换为char后访问的是一个字节。
}
int main()
{
int ret = check_sys();
if (ret == 1)
{
printf("小端\n");
}
else
{
printf("大端\n");
}
return 0;
}
总结:涉及到内存访问就要考虑到指针并且注意指针变量的类型。