大端小端的概念(以下概念来自网络)
端模式(Endian)的这个词出自Jonathan Swift书写的《格列佛游记》。这本书根据将鸡蛋敲开的方法不同将所有的人分为两类,从圆头开始将鸡蛋敲开的人被归为Big Endian,从尖头开始将鸡蛋敲开的人被归为Littile Endian。小人国的内战就源于吃鸡蛋时是究竟从大头(Big-Endian)敲开还是从小头(Little-Endian)敲开。在计算机业Big Endian和Little Endian也几乎引起一场战争。在计算机业界,Endian表示数据在存储器中的存放顺序。下文举例说明在计算机中大小端模式的区别。
如果将一个32位的整数0x12345678存放到一个整型变量(int)中,这个整型变量采用大端或者小端模式在内存中的存储由下表所示。为简单起见,本书使用OP0表示一个32位数据的最高字节MSB(Most Significant Byte),使用OP3表示一个32位数据最低字节LSB(Least Significant Byte)。
地址偏移 | 大端模式 | 小端模式 |
0x00 | 12(OP0) | 78(OP3) |
0x01 | 34(OP1) | 56(OP2) |
0x02 | 56(OP2) | 34(OP1) |
0x03 | 78(OP3) | 12(OP0) |
如果将一个16位的整数0x1234存放到一个短整型变量(short)中。这个短整型变量在内存中的存储在大小端模式由下表所示。
地址偏移 | 大端模式 | 小端模式 |
0x00 | 12(OP0) | 34(OP1) |
0x01 | 34(OP1) | 12(OP0) |
由上表所知,采用大小模式对数据进行存放的主要区别在于在存放的字节顺序,大端方式将高位存放在低地址,小端方式将低位存放在高地址。采用大端方式 进行数据存放符合人类的正常思维,而采用小端方式进行数据存放利于计算机处理。到目前为止,采用大端或者小端进行数据存放,其孰优孰劣也没有定论。
C语言中判断大小端的方法
有时候,用C语言写程序时需要知道是大端模式还是小端模式。 所谓的大端模式,是指数据的低位保存在内存的高地址中,而数据的高位,保存在内存的低地址中;小端模式相反。为什么会有大小端模式之分呢?这是因为在计算机系统中,我们是以字节为单位的,每个地址单元都对应着一个字节,一个字节为8bit。但是在C语言中除了8bit的char之外,还有16bit的short型,32bit的long型(要看具体的编译器),另外,对于位数大于8位的处理器,例如16位或者32位的处理器,由于寄存器宽度大于一个字节,那么必然存在着一个如果将多个字节安排的问题。因此就导致了大端存储模式和小端存储模式。例如一个16bit的short型x,在内存中的地址为0x0010,x的值为0x1122,那么0x11为高字节,0x22为低字节。对于大端模式,就将0x11放在低地址中,即0x0010中,0x22放在高地址中,即0x0011中。小端模式,刚好相反。我们常用的X86结构是小端模式,而KEIL-C51则为大端模式。很多的ARM,DSP都为小端模式。有些ARM处理器还可以由硬件来选择是大端模式还是小端模式。
下面这段代码可以用来测试一下你的编译器是大端模式还是小端模式:
点击(此处)折叠或打开
- #include <stdio.h>
-
- int main()
- {
- short int x;
- char x1,x2;
- x = 0x1122;
- x1 = ((char *)&x)[0]; //低地址
- x2 = ((char *)&x)[1]; //高地址
- printf("x1=%x\n",x1);
- printf("x2=%x\n",x2);
- return 0;
- }
若x1=0x11,则是大端; 若x2=0x22,则是小端......
点击(此处)折叠或打开
- #include<stdio.h>
-
- int main()
- {
- int x = 1;
- char *p = (char *)&x;
-
- if(*p)
- {
- printf("little\n");
- }
- else
- {
- printf("large\n");
- }
- return 0;
- }
存储器模式(SMALL LARGE COMPACT)
C51 提供了3 种存储器模式来存储变量过程参数和分配再入函数堆栈你应该尽量
使用小存储器模式,很少应用系统需要使用其它两种模式。像有大的再入函数堆栈系统那
样一般来说如果系统所需要的内存数小于内部RAM 数时都应以小存储模式进行编译,
在这种模式下DATA 段是所有内部变量和全局变量的默认存储段所有参数传递都发生在
DATA 段中。如果有函数被声明为再入函数编译器会在内部RAM 中为他们分配空间,这种模式的优势就是数据的存取速度很快,但只有120 个字节的存储空间供你使用总共有128个字节,但至少有8 个字节被寄存器组使用你还要为程序调用开辟足够的堆栈
如果你的系统有256 字节或更少的外部RAM 你可以使用压缩存储模式,这样一来
如果不加说明,变量将被分配在PDATA 段中,这种模式将扩充你能够使用的RAM 数量对XDATA 段以外的数据存储仍然是很快的变量的参数传递,将在内部RAM 中进行,这样存储速度会比较快,对PDATA 段的数据的寻址是通过R0 和R1 进行间接寻址比使用DPTR 要快一些,在大存储模式中所有变量的默认存储区是XDATA 段,Keil C 尽量使用内部寄存器组进行参数传递,在寄存器组中可以传递参数的数量和和压缩存储模式一样再入函数的模拟栈将在XDATA 中,对XDATA 段数据的访问是最慢的,所以要仔细考虑变量应存储的位置使数据的存储速度得到优化。