ARM存储格式之大端小端

出处:维库电子市场网 发布于:2016-08-15 15:36:25

   开头讲个有关 大端小端的故事:

  端模式(Endian)的这个词出自Jonathan Swift书写的《格列佛游记》。这本书根据将鸡蛋敲开的方法不同将所有的人分为两类,从圆头开始将鸡蛋敲开的人被归为Big Endian,从尖头开始将鸡蛋敲开的人被归为Littile Endian。小人国的内战就源于吃鸡蛋时是究竟从大头(Big-Endian)敲开还是从小头(Little-Endian)敲开。在计算机业Big Endian和Little Endian也几乎引起一场战争。

  我们知道在内存中数据是以字节为单位进行存储的,每个地址单元对应着一个字节(byte),一个字节为8位(bite)。但是很多时候数据除了8bit额char外,还有16bit的short,32位的long型(要看具体的编译器),必然存在多字节安排的问题。不同的计算机存放多字节值的顺序不同,有些机器在起始地址存放低位字节(低位先存),即小端模式;有的机器在起始地址存放高位字节(高位先存),即大端模式。基于Intel的CPU,采用的是低位先存。而KEIL C51则为大端模式。大端小端对应着数据在存储器中的存放顺序。

  同时,在网络传输中,网络协议需要指定网络字节顺序,TCP/IP协议中使用16位整数和32位整数的高位先存模式,对应我们的大端模式。

  下面是两个具体例子:

  16bit宽的数0x1234在Little-endian模式(以及Big-endian模式)CPU内存中的存放方式(假设从地址0x4000开始存放)为:

ARM存储格式之大端小端

  32bit宽的数0x12345678在Little-endian模式以及Big-endian模式)CPU内存中的存放方式(假设从地址0x4000开始存放)为:

ARM存储格式之大端小端

  联合体union的存放顺序是所有成员都从低地址开始存放,利用该特性可以轻松地获得了CPU对内存采用Little-endian还是Big-endian模式读写。

  写程序判断处理器是Little-endian模式,还是Big-endian模式,可以通过以下程序:

  1、通过将int强制类型转换成char单字节,通过判断起始存储位置。

  1 void main(int argc, char **argv)

  2 {

  3 int i = 1;

  4 char *cp = (char *)&i; //前面是指针运算符*,前值类型转换。后面是取地址符号。

  5 if (*cp) //如果此时cp指向的内存为1的话,则为小端,否则为大端。

  6 printf("Little Endian\n");

  7 else

  8 printf("Big Endian\n");

  9

  10 exit(EXIT_SUCCESS);

  11 }

  注释:如果小端方式中(i占至少两个字节的长度)则i所分配的内存地址那个字节中就存着1,其他字节是0.大端的话则1在i的地址字节处存放,char是一个字节,所以强制将char型量p指向i则p指向的一定是i的地址,那么就可以判断p中的值是不是1(或者为0,也即是假)来确定是不是小端。

或者如下程序:

  void main()

  {

  short int x;

  char x0,x1;

  x=0x1122;

  x0=((char*)&x)[0]; //低地址单元

  x1=((char*)&x)[1]; //高地址单元

  if (0x11 == x0 && 0x22 == x1)

  {

  cout << "Big_endian" << endl;

  }

  else

  {

  cout << "Little_endian" << endl;

  }

  }

  2、利用联合体union的存放顺序是所有成员都从低地址开始存放,判断处理器模式。

  bool checkCPU( )

  {

  {

  union w

  {

  int a;

  char b;

  } c;

  c.a = 1;

  return(c.b ==1);

  }

  }

  以及如下程序:

  bool isLittleEndian()

  {

  union _dword

  {

  int all;

  struct _bytes

  {

  char byte0;

  char pad[3];

  }bytes;

  }dword;

  dword.all=0x87654321;

  return (0x21==dword.bytes.byte0);

  }

  分析:如果你的处理器调用函数isLittleEndian返回1,那么说明你的处理器为little endian,否则为big endian.注意,如果在little endian处理器上,byte0和pad按内存从低到高的存放顺序:LOW->byte0 pad[0] pad[1] pad[2] ->HIGH;0x87654321按内存从低到高的存放顺序: 0x21 0x43 0x65 0x87, 可见byte0对应到0x21。所以通过判断dword中个字节dword.bytes.byte0是否与0x21相等就可以看出是否是little endian。

 

关键词:ARM存储

版权与免责声明

凡本网注明“出处:维库电子市场网”的所有作品,版权均属于维库电子市场网,转载请必须注明维库电子市场网,https://www.dzsc.com,违反者本网将追究相关法律责任。

本网转载并注明自其它出处的作品,目的在于传递更多信息,并不代表本网赞同其观点或证实其内容的真实性,不承担此类作品侵权行为的直接责任及连带责任。其他媒体、网站或个人从本网转载时,必须保留本网注明的作品出处,并自负版权等法律责任。

如涉及作品内容、版权等问题,请在作品发表之日起一周内与本网联系,否则视为放弃相关权利。

广告
上传BOM文件: BOM文件
*公司名:
*联系人:
*手机号码:
QQ:
应用领域:

有效期:
OEM清单文件: OEM清单文件
*公司名:
*联系人:
*手机号码:
QQ:
有效期:

扫码下载APP,
一键连接广大的电子世界。

在线人工客服

买家服务:
卖家服务:

0571-85317607

客服在线时间周一至周五
9:00-17:30

关注官方微信号,
第一时间获取资讯。

建议反馈

联系人:

联系方式:

按住滑块,拖拽到最右边
>>
感谢您向阿库提出的宝贵意见,您的参与是维库提升服务的动力!意见一经采纳,将有感恩红包奉上哦!