关键是big endian和little
endian的概念。注意16bit机器和32bit机器(以32位为访问单位)下排列不同。 内存地址从低地址向高地址增长,big endian是高位数据优先,即高位放在低地址。而little endian是低位数据优先,低位放在低地址。网络字节序采用大端。 判断字节序: main() { if ((*(char *)&x) == 0x1) printf("little endian!\n"); else printf("big endian!\n"); } 字节序转换: ntohs(n) = __swab16(n), ntohl = __swab32(n)。 __swab16与__swab32函数定义如下所示。 #define ___swab16(x) { __u16 __x = (x); ((__u16)( (((__u16)(__x) & (__u16)0x00ffU) << 8) | (((__u16)(__x) & (__u16)0xff00U) >> 8) )); } #define ___swab32(x) { __u32 __x = (x); ((__u32)( (((__u32)(__x) & (__u32)0x000000ffUL) << 24) | (((__u32)(__x) & (__u32)0x0000ff00UL) << 8) | (((__u32)(__x) & (__u32)0x00ff0000UL) >> 8) | (((__u32)(__x) & (__u32)0xff000000UL) >> 24) )); } 写法很严谨。简化版的: #define swab32(x) ((x&0x000000ff) << 24 | (x&0x0000ff00) << 8 | (x&0x00ff0000) >> 8 | (x&0xff000000) >> 24) #define swab16(x) ((x&0x00ff) << 8 | (x&0xff00) >> 8) |
|