大端模式小端模式

来源:互联网 发布:php好吗 编辑:程序博客网 时间:2024/05/29 03:00

作者: 李云鹏(qqliyunpeng@sina.cn) 

版本号: 20170330 
更新时间: <2017-04-06> 
原创时间: <2017-03-30> 


http://blog.csdn.net/qqliyunpeng/article/details/68484497


1. 概念简介


        不同的系统在存储数据时是分大端(bit-endian)小端(little-endian)存储的,比如,Inter x86、ARM核采用的是小端模式,Power PC、MIPS UNIX和HP-PA UNIX采用大端模式

小端模式用文字描述是,低地址上存放低字节,高地址上存放高字节。

假如有一个32位的数据 0x11223344,则在小端模式上的机器上存储为如下的形式:

【1】0x11223344这个数中 0x11 是高字节(MSB),0x44是地字节(LSB)

【2】讨论大小端的时候最小单位是字节

【3】内存的画法中采用的是向上增长的

【3】可以将数据比作方向盘,顺时钟旋转得到的在内存中的布局是小端存储

至于大端模式用文字描述是,低地址上存放高字节,高地址上存放低字节。


2. 如何判断


判断的方法有很多种,下面将简单的列举几种:

第一种方法:

[cpp] view plain copy
 print?
  1. /* 
  2.  * 1: little-endian 
  3.  * 0: big-endian 
  4.  */  
  5. int checkEndian()  
  6. {  
  7.     int a = 1;  
  8.     char *p = (char *)&a;  
  9.   
  10.     return (*p == 1);  
  11. }  
【1】如果是大端,*p的结果是0

第二种方法:

[cpp] view plain copy
 print?
  1. /* 
  2.  * 1: little-endian 
  3.  * 0: big-endian 
  4.  */  
  5. int checkEndian()  
  6. {  
  7.     union w  
  8.     {  
  9.         int a;  
  10.         char b;  
  11.     } c;  
  12.     c.a = 1;  
  13.     return (c.b == 1);  
  14. }  


函数中打印方法:

[cpp] view plain copy
 print?
  1. printf("%s\n", checkEndian() ? "little-endian" : "big-endian");  

3. 大端和小端的转换


[cpp] view plain copy
 print?
  1. int big_litle_endian(int x)    
  2. {    
  3.     int tmp;    
  4.     tmp = (((x)&0xff)<<24) + (((x>>8)&0xff)<<16) + (((x>>16)&0xff)<<8) + (((x>>24)&0xff));    
  5.     return tmp;    
  6. }  

4. 其他


1. 在通信的场合经常会遇到大端和小端的转换的问题,比如tcp/ip 中,tcp/ip 中规定了自己传输的时候采用大端模式,当然相应的它也提供了很多函数来做支持。

如果主机是小端的模式,在跟网络进行交互的时候经常要用到如下的函数

  • htons —— 把unsigned short类型从 主机序 转成 网络字节序
  • ntohs —— 把unsigned short类型从 网络字节序 转成 主机序
  • htonl —— 把unsigned long类型从 主机序 转成 网络字节序
  • ntohl —— 把unsigned long类型从 网络字节序 转成 主机序
需要包含头文件
[cpp] view plain copy
 print?
  1. #if defined(_LINUX) || defined(_DARWIN)  
  2. #include <netinet/in.h>  
  3. #endif  
  4.   
  5. #ifdef WIN32  
  6. #include <WINSOCK2.H>  
  7. #endif  

当一个系统要发送的数据是 0x12345678,以大端模式发送,则会先发送0x12.


2. 如何在64位ubuntu下同下编译32位的程序?

需要先安装32位的库:sudo apt-get install libc6-dev-i386

然后在编译的时候加上-m32选项。


原创粉丝点击