大端小端(Big- Endian和Little-Endian)

字节序(Endian),大端(Big-Endian),小端(Little-Endian) 图文并茂html

http://www.cppblog.com/tx7do/archive/2009/01/06/71276.html编程

 

在各类计算机体系结构中,对于字节、字等的存储机制有所不一样,于是引起了计算机通讯领 域中一个很重要的问题,即通讯双方交流的信息单元(比特、字节、字、双字等等)应该以什么样的顺序进行传送。若是不达成一致的规则,通讯双方将没法进行正 确的编/译码从而致使通讯失败。目前在各类体系的计算机中一般采用的字节存储机制主要有两种:Big-Endian和Little-Endian,下面先 从字节序提及。数组

 

1、什么是字节序网络

字节序,顾名思义字节的顺序,再多说两句就是大于一个字节类型的数据在内存中的存放顺序(一个字节的数据固然就无需谈顺序的问题了)其实大部分人在实际的开 发中都不多会直接和字节序打交道。惟有在跨平台以及网络程序中字节序才是一个应该被考虑的问题。函数

在全部的介绍字节序的文章中都会提到字 节序分为两类:Big-Endian和Little-Endian,引用标准的Big-Endian和Little-Endian的定义以下:
a) Little-Endian就是低位字节排放在内存的低地址端,高位字节排放在内存的高地址端
b) Big-Endian就是高位字节排放在内存的低地址端,低位字节排放在内存的高地址端
c) 网络字节序:TCP/IP各层协议将字节序定义为Big-Endian,所以TCP/IP协议中使用的字节序一般称之为网络字节序。
布局

1.1 什么是高/低地址端post

首先咱们要知道咱们C程序映像中内存的空间布局状况:在《C专 家编程》中或者《Unix环境高级编程》中有关于内存空间布局状况的说明,大体以下图:
----------------------- 最高内存地址 0xffffffff
栈底

栈顶

-----------------------
ui

NULL (空洞)
-----------------------

-----------------------
未初始 化的数据
----------------------- 统称数据段
初始化的数据
-----------------------
正 文段(代码段)
----------------------- 最低内存地址 0x00000000
编码

以上图为例若是咱们在栈 上分配一个unsigned char buf[4],那么这个数组变量在栈上是如何布局的呢?看下图:
栈底 (高地址)
----------
buf[3]
buf[2]
buf[1]
buf[0]

----------
栈顶 (低地址)
spa

1.2 什么是高/低字节

如今咱们弄清了高/低地址,接着考虑高/低字节。有些文章中称低位字节为最低有效位,高位字节为最高有效位。若是咱们有一个32位无符号整型0x12345678,那么高位是什么,低位又是什么呢? 其实很简单。在十进制中咱们都说靠左边的是高位,靠右边的是低位,在其余进制也是如此。就拿 0x12345678来讲,从高位到低位的字节依次是0x十二、0x3四、0x56和0x78
高/低地址端和高/低字节都弄清了。咱们再来回顾 一下Big-Endian和Little-Endian的定义,并用图示说明两种字节序:
以unsigned int value = 0x12345678为例,分别看看在两种字节序下其存储状况,咱们能够用unsigned char buf[4]来表示value:

Big-Endian: 低地址存放高位,以下图:
栈底 (高地址)
---------------
buf[3] (0x78) -- 低位
buf[2] (0x56)
buf[1] (0x34)
buf[0] (0x12) -- 高位
---------------
栈顶 (低地址)

Little-Endian: 低地址存放低位,以下图:
栈底 (高地址)
---------------
buf[3] (0x12) -- 高位
buf[2] (0x34)
buf[1] (0x56)
buf[0] (0x78) -- 低位
--------------
栈 顶 (低地址)

 

2、各类Endian

2.1 Big-Endian

计算机体系结构中一种描述多字节存储顺序的术语,在这种机制中最重要字节(MSB)存放在最低端的地址 上。采用这种机制的处理器有IBM3700系列、PDP-十、Mortolora微处理器系列和绝大多数的RISC处理器。
+----------+
| 0x34 |<-- 0x00000021
+----------+
| 0x12 |<-- 0x00000020
+----------+
图 1:双字节数0x1234以Big-Endian的方式存在起始地址0x00000020中

 在Big-Endian中,对于bit序列 中的序号编排方式以下(以双字节数0x8B8A为例):
bit 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
+-----------------------------------------+
val | 1 0 0 0 1 0 1 1 | 1 0 0 0 1 0 1 0 |
+----------------------------------------+
图 2:Big-Endian的bit序列编码方式

2.2 Little-Endian

计算机体系结构中 一种描述多字节存储顺序的术语,在这种机制中最不重要字节(LSB)存放在最低端的地址上。采用这种机制的处理器有PDP-十一、VAX、Intel系列 微处理器和一些网络通讯设备。该术语除了描述多字节存储顺序外还经常用来描述一个字节中各个比特的排放次序。

+----------+
| 0x12 |<-- 0x00000021
+----------+
| 0x34 |<-- 0x00000020
+----------+

图3:双字节数0x1234以Little-Endian的方式存在起始地址0x00000020中

 在 Little-Endian中,对于bit序列中的序号编排和Big-Endian恰好相反,其方式以下(以双字节数0x8B8A为例):

bit 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
+-----------------------------------------+
val | 1 0 0 0 1 0 1 1 | 1 0 0 0 1 0 1 0 |
+-----------------------------------------+
图 4:Little-Endian的bit序列编码方式

注2:一般咱们说的主机序(Host Order)就是遵循Little-Endian规则。因此当两台主机之间要经过TCP/IP协议进行通讯的时候就须要调用相应的函数进行主机序 (Little-Endian)和网络序(Big-Endian)的转换

注3:正由于这两种机制对于同一bit序列的序号编排方式恰 恰相反,因此《现代英汉词典》中对MSB的翻译为“最高有效位”欠妥,故本文定义为“最重要的bit/byte”。

2.3 Middle-Endian

除了Big-Endian和Little-Endian以外的多字节存储顺序就是Middle- Endian,好比以4个字节为例:象以3-4-1-2或者2-1-4-3这样的顺序存储的就是Middle-Endian。这种存储顺序偶尔会在一些小 型机体系中的十进制数的压缩格式中出现

嵌入式系统开发者应该对Little-endian和Big-endian模式很是了解。采用 Little-endian模式的CPU对操做数的存放方式是从低字节到高字节,而Big-endian模式对操做数的存放方式是从高字节到低字节。 32bit宽的数0x12345678在Little-endian模式CPU内存中的存放方式(假设从地址0x4000开始存放)为:

内存 地址 0x4000 0x4001 0x4002 0x4003
存放内容 0x78 0x56 0x34 0x12

 

而在Big- endian模式CPU内存中的存放方式则为:

内存地址 0x4000 0x4001 0x4002 0x4003
存放内容 0x12 0x34 0x56 0x78

 

3、Big-EndianLittle-Endian优缺点

Big-Endian优势:靠首先提取高位字节,你老是能够由看看在偏移位置为0的字节来肯定这个数字是 正数仍是负数。你没必要知道这个数值有多长,或者你也没必要过一些字节来看这个数值是否含有符号位这个数值是以它们被打印出来的顺序存放的,因此从二进制到十进制的函数特别有效。于是,对于不一样要求的机器,在设计存取方式时就会不一样。

Little-Endian优势:提取一个,两个,四个或者更长字节数据的汇编指令以与其余全部格式相同的方式进行:首先在偏移地址为0的地方提取最低位的字节,由于地址偏移和字节数是一对 一的关系,多重精度的数学函数就相对地容易写了

若是你增长数字的值,你可能在左边增长数字(高位非指数函数须要更多的数字)。所以, 常常须要增长两位数字并移动存储器里全部Big-endian顺序的数字,把全部数向右移,这会增长计算机的工做量。不过,使用Little- Endian的存储器中不重要的字节能够存在它原来的位置,新的数能够存在它的右边的高位地址里。这就意味着计算机中的某些计算能够变得更加简单和快速。

4、如何检查处理器是Big-Endian仍是Little-Endian?

因为联合体union的存放顺序是全部成员都从低地址开始存放,利用该特性就能够轻松地得到了CPU对内存采用Little- endian仍是Big-endian模式读写。例如:
int checkCPUendian(){
union {
unsigned int a;
unsigned char b;           
}c;
c.a = 1;
return (c.b == 1);      

}   /*return 1 : little-endian, return 0:big-endian*/

 

5、Big-EndianLittle-Endian转 换

 

现有的平台上Intel的X86采用的是Little-Endian,而像 Sun的SPARC采用的就是Big-Endian。那么在跨平台或网络程序中如何实现字节序的转换呢?这个经过C语言的移位操做很容易实现,例以下面的 宏:

#if defined(BIG_ENDIAN) && !defined(LITTLE_ENDIAN)

#define htons(A)   (A)
#define htonl(A)     (A)
#define ntohs(A)   (A)
#define ntohl(A)    (A)

#elif defined(LITTLE_ENDIAN) && !defined(BIG_ENDIAN)

#define htons(A)     ((((uint16)(A) & 0xff00) >> 8) | \
(((uint16)(A) & 0x00ff) << 8))
#define htonl(A)     ((((uint32)(A) & 0xff000000) >> 24) | \
(((uint32)(A) & 0x00ff0000) >> 8) | \
(((uint32)(A) & 0x0000ff00) << 8) | \
(((uint32)(A) & 0x000000ff) << 24))
#define ntohs htons
#define ntohl htohl

#else

#error "Either BIG_ENDIAN or LITTLE_ENDIAN must be #defined, but not both."

 

 

网络字节顺序
一、字节内的比特位不受这种顺序的影响
好比一个字节 1000 0000 (或表示为十六进制 80H)不论是什么顺序其内存中的表示法都是这样。

 

二、大于1个字节的数据类型才有字节顺序问题
好比 Byte A,这个变量只有一个字节的长度,因此根据上一条没有字节顺序问题。因此字节顺序是“字节之间的相对顺序”的意思。


三、大于1个字节的数据类型的字节顺序有两种
好比 short B,这是一个两字节的数据类型,这时就有字节之间的相对顺序问题了。
网络字节顺序是“所见即所得”的顺序。而Intel类型的CPU的字节顺序与此相反。
好比上面的 short B=0102H(十六进制,每两位表示一个字节的宽度)。所见到的是“0102”,按通常数学常识,数轴从左到右的方向增长,即内存地址从左到右增长的话,在内存中这个 short B的字节顺序是:
01 02
这就是网络字节顺序。所见到的顺序和在内存中的顺序是一致的!
假设经过抓包获得网络数据的两个字节流为:01 02

而相反的字节顺序就不一样了,其在内存中的顺序为:02 01

若是这表示两个 Byte类型的变量,那么天然不须要考虑字节顺序的问题。若是这表示一个 short 变量,那么就须要考虑字节顺序问题。根据网络字节顺序“所见即所得”的规则,这个变量的值就是:0102

假设本地主机是Intel类型的,那么要表示这个变量,有点麻烦:
定义变量 short X,字节流地址为:pt,按顺序读取内存是为x=*((short*)pt);
那么X的内存顺序固然是 01 02按非“所见即所得”的规则,这个内存顺序和看到的同样显然是不对的,因此要把这两个字节的位置调换。调换的方法能够本身定义,但用已经有的API仍是更为方便。

网络字节顺序与主机字节顺序
NBO与HBO 网络字节顺序NBO(Network Byte Order):按从高到低的顺序存储,在网络上使用统一的网络字节顺序,能够避免兼容性问题。主机字节顺序(HBO,Host Byte Order):不一样的机器HBO不相同,与CPU设计有关计算机数据存储有两种字节优先顺序:高位字节优先和低位字节优先Internet上数据以高位字节优先顺序在网络上传输,因此对于在内部是以低位字节优先方式存储数据的机器,在Internet上传输数据时就须要进行转换。

htonl()
简述:
    将主机的无符号长整形数转换成网络字节顺序。
    #include <winsock.h>
    u_long PASCAL FAR htonl( u_long hostlong);
    hostlong:主机字节顺序表达的32位数。
注释:
    本函数将一个32位数从主机字节顺序转换成网络字节顺序。
返回值:
    htonl()返回一个网络字节顺序的值。

inet_ntoa()
简述:
将网络地址转换成“.”点隔的字符串格式。
#include <winsock.h>
char FAR* PASCAL FAR inet_ntoa( struct in_addr in);
in:一个表示Internet主机地址的结构。
注释:
本函数将一个用in参数所表示的Internet地址结构转换成以“.” 间隔的诸如“a.b.c.d”的字符串形式。请注意inet_ntoa()返回的字符串存放在WINDOWS套接口实现所分配的内存中。应用程序不该假设该内存是如何分配的。在同一个线程的下一个WINDOWS套接口调用前,数据将保证是有效。
返回值:
若无错误发生,inet_ntoa()返回一个字符指针。不然的话,返回NULL。其中的数据应在下一个WINDOWS套接口调用前复制出来。

网络中传输的数据有的和本地字节存储顺序一致,而有的则大相径庭,为了数据的一致性,就要把本地的数据转换成网络上使用的格式,而后发送出去,接收的时候也是同样的,通过转换而后才去使用这些数据,基本的库函数中提供了这样的能够进行字节转换的函数,如和htons( ) htonl( ) ntohs( ) ntohl( ),这里n表示network,h表示host,htons( ) htonl( )用于本地字节向网络字节转换的场合,s表示short,即对2字节操做,l表示long即对4字节操做。一样ntohs( )ntohl( )用于网络字节向本地格式转换的场合。

 

 

#endif

相关文章
相关标签/搜索