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

基本概念

首先来看某百科定义:

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

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

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

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

### 回答1: 大小端指的是数据在内存中存储的顺序,大端模式是高位字节存储在低地址,小端模式是高位字节存储在高地址。在Linux下,可以使用以下代码判断当前机器的大小端模式: ``` #include <stdio.h> int main() { unsigned int x = 0x12345678; char *c = (char*)&x; if (*c == 0x12) { printf("Big Endian\n"); } else { printf("Little Endian\n"); } return 0; } ``` ### 回答2: 在Linux中,判断系统是大端还是小端可以通过以下方法: 1. 使用C语言的方式判断:可以使用联合体的方式来判断大小端。可以创建一个包含一个char类型和一个int类型的联合体,然后将这个联合体的char和int成员的值分别设置为1,然后通过判断char成员的值来确定系统的字节序。若为1,则为小端;若为0,则为大端。 ```c #include <stdio.h> int main() { union { char c; int i; } endian; endian.i = 1; if (endian.c == 1) { printf("小端字节序\n"); } else { printf("大端字节序\n"); } return 0; } ``` 2. 使用命令行工具:在Linux终端中使用命令`lscpu`可以获取CPU的详细信息,其中有一项是"Byte Order",如果显示为"Little Endian"则表示是小端字节序,如果显示为"Big Endian"则表示是大端字节序。 综上所述,以上两种方法都可以在Linux中判断大小端。 ### 回答3: 在Linux中,可以通过以下方法判断当前系统的字节序是大端(Big Endian)还是小端(Little Endian)。 方法一:使用C语言编写一个小程序进行判断。 ```c #include<stdio.h> int main() { unsigned int num = 1; char *p = (char *)&num; if (*p == 1) { printf("该系统为小端字节序\n"); } else { printf("该系统为大端字节序\n"); } return 0; } ``` 该程序定义了一个无符号整型变量`num`并给其赋值为1,然后通过指针`p`来指向`num`的地址。由于整型在内存中是以字节的形式存储的,我们将首地址的字节与1进行比较,如果相等,则说明是小端字节序,否则为大端字节序。 方法二:使用命令行查看系统的字节序。 在Linux中,我们可以通过命令`lscpu`来查看系统的相关信息,其中包括字节序。在终端中输入如下命令: ``` lscpu | grep "Byte Order" ``` 如果输出结果为`Byte Order: Little Endian`,则说明系统为小端字节序。如果输出结果为`Byte Order: Big Endian`,则说明系统为大端字节序。 以上两种方法可以帮助我们在Linux中判断大小端
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值