大小端模式以及两种判断方法

基本概念

首先来看某百科定义:

  • 大端模式,是指数据的高字节保存在内存的低地址中,而数据的低字节保存在内存的高地址中。
  • 小端模式,是指数据的高字节保存在内存的高地址中,而数据的低字节保存在内存的低地址中。

这里我们看到其实大端小端就是数据在内存中的存放的字节顺序
其实上面的定义有点儿难以记忆,总感觉很绕,一个我觉得高效的记忆方式就是:

  • 大端:高尾端:数据的尾部(低位字节)放在内存的高位地址。
  • 小端:低尾端:数据的尾部(低位字节)放在内存的地位地址。

之所以出现这样的问题,是因为一个字节是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");
} 

END

  • 4
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值