基本概念
首先来看某百科定义:
- 大端模式,是指数据的高字节保存在内存的低地址中,而数据的低字节保存在内存的高地址中。
- 小端模式,是指数据的高字节保存在内存的高地址中,而数据的低字节保存在内存的低地址中。
这里我们看到其实大端小端就是数据在内存中的存放的字节顺序。
其实上面的定义有点儿难以记忆,总感觉很绕,一个我觉得高效的记忆方式就是:
- 大端:高尾端:数据的尾部(低位字节)放在内存的高位地址。
- 小端:低尾端:数据的尾部(低位字节)放在内存的地位地址。
之所以出现这样的问题,是因为一个字节是8位,而现在CPU中的寄存器的位数是大于8的(一般来讲,多少位的处理器寄存器就多少位),所以就存在一个从内存中读取数据到寄存器的顺序问题。而不同处理器读取内存的方式不同,所以不同的架构的内存存放也有所不同。我们常用的X86结构是小端模式,而KEIL C51则为大端模式。有些ARM处理器还可以由硬件来选择是大端模式还是小端模式。
下面拿一个例子来说明:
假设我们现在有int型变量 0x12345678:,那么如果是小端模式,即低尾端,数据的低位0x78在内存的低位地址中存放。其在内存的存放方式如下图所示:
若是大端模式,即高尾端,那么数据的低位0x78存在内存中的高位地址。其内存存放方式如下:
判断方法
首先,要强调下不能用移位,位与等运算方式来判断大小端模式:
因为前面提到过,大小端是数据在内存中的字节存放顺序,而运算指令是由CPU执行的,也就是说运算等方法其实是改变的CPU中寄存器的值,而寄存器中的值通过大小端模式已经得到了正确的数据存放顺序。
- 判断方法一:利用联合类型判断
union类型的主要特点如下:- union中可以定义多个成员,union的大小由最大的成员的大小决定;
- union成员共享同一块大小的内存,一次只能使用其中的一个成员;
- 对某一个成员赋值,会覆盖其他成员的值;
- 联合体union的存放顺序是所有成员都从低地址开始存放。
所以我们可以定义联合体如下:
//method 1
union bit{//对齐原则,char与int指向的均是低位地址
int a;
char b;
};
这个时候我们赋值 a = 0x12345678,如果低位字节的b存放的是0x78,则说明是小端模式,若为0x12则为大端模式:
#include <stdio.h>
union bit{//对齐原则,char与int指向的均是低位地址
int a;
char b;
};
int main(){
bit test;
test.a = 0x12345678;
if(test.b == 0x78)//如果低位地址保存的是1,即低位字节在低位地址,为小端
printf("本机为小端模式\n");
else
printf("本机为大端模式\n");
return 0;
}
- 判断方法二:利用强制类型转换判断
这种方式需要定义一个字节指针,指向int型的低位地址,因为要用到强制类型转换,故而称为利用强制类型转换的判断方法:
//method 2
void judge(void){
int i = 0x12345678;
char *p;
p = (char *)&i;//强制类型转换成char*型指针,指向的位置为低位地址
if(*p == 0x78)//低位地址存储的是低位字节,则为小端
printf("本机为小端模式\n");
else
printf("本机为大端模式\n");
}