1、大端和小端的问题网络
对于整型、长整型、无符号整型等数据类型,Big endian 认为第一个字节是最高位字节(按照从低地址到高地址的顺序存放数据的高位字节到低位字节);而 Little endian 则相反,它认为第一个字节是最低位字节(按照从低地址到高地址的顺序存放据的低位字节到高位字节)。ui
例如,假设从内存地址 0x0000 开始有如下数据:
0x0000 0x0001 0x0002 0x0003
0x12 0x34 0xab 0xcd
若是咱们去读取一个地址为 0x0000 的四个字节变量,若字节序为big-endian,则读出结果为0x1234abcd;若字节序为little-endian,则读出结果为0xcdab3412。spa
若是咱们将0x1234abcd 写入到以 0x0000 开始的内存中,则Little endian 和 Big endian 模式的存放结果以下:
地址 0x0000 0x0001 0x0002 0x0003
big-endian 0x12 0x34 0xab 0xcd
little-endian 0xcd 0xab 0x34 0x12操作系统
通常来讲,x86 系列 CPU 都是 little-endian 的字节序,PowerPC 一般是 big-endian,网络字节顺序也是 big-endian还有的CPU 能经过跳线来设置 CPU 工做于 Little endian 仍是 Big endian 模式。code
对于0x12345678的存储:blog
小端模式:(从低字节到高字节)
地位地址 0x78 0x56 0x34 0x12 高位地址内存
大端模式:(从高字节到低字节)
地位地址 0x12 0x34 0x56 0x78 高位地址it
2、大端小端转换方法io
htonl() htons() 从主机字节顺序转换成网络字节顺序
ntohl() ntohs() 从网络字节顺序转换为主机字节顺序class
Big-Endian转换成Little-Endian
#define BigtoLittle16(A) ((((uint16)(A) & 0xff00) >> 8) | (((uint16)(A) & 0x00ff) << 8)) #define BigtoLittle32(A) ((((uint32)(A) & 0xff000000) >> 24) | (((uint32)(A) & 0x00ff0000) >> 8) | \ (((uint32)(A) & 0x0000ff00) << 8) | (((uint32)(A) & 0x000000ff) << 24))
3、大端小端检测方法
如何检查处理器是big-endian仍是little-endian?
C程序:
int i = 1; char *p = (char *)&i; if(*p == 1) printf("Little Endian"); else printf("Big Endian");
大小端存储问题,若是小端方式中(i占至少两个字节的长度)则i所分配的内存最小地址那个字节中就存着1,其余字节是0.大端的话则1在i的最高地址字节处存放,char是一个字节,因此强制将char型量p指向i则p指向的必定是i的最低地址,那么就能够判断p中的值是否是1来肯定是否是小端。
联合体union的存放顺序是全部成员都从低地址开始存放,利用该特性就能够轻松地得到了CPU对内存采用Little-endian仍是Big-endian模式读写
/*return 1: little-endian, return 0: big-endian*/ int checkCPUendian() { union { unsigned int a; unsigned char b; }c; c.a = 1; return (c.b == 1); }
实现一样的功能,来看看Linux 操做系统中相关的源代码是怎么作的:
static union { char c[4]; unsigned long mylong; } endian_test = {{ 'l', '?', '?', 'b' } }; #define ENDIANNESS ((char)endian_test.mylong)