我们知道,网络字节序是TCP/IP各层协议将字节序定义为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
- 判断是大端还是小端
- 小端向大端的转换
1. 判断是大端还是小端
思路:将int类型转换为char类型,判断最低字节。
int main(void)
{
short int a = 0x1234;
char *p = (char *)&a;
if( *p == 0x34)
cout<<"Little endian"<<endl;
else
cout<<"Big endian"<<endl;
return 0;
}
int main(void)
{
short int a = 0x1234;
char x0 , x1;
x0 = ((char *)&a)[0];
x1 = ((char *)&a)[1];
//小端:低地址存放低字节
if( x0 == 0x34)
cout<<"Little endian"<<endl;
else
cout<<"Big endian"<<endl;
return 0;
}
2. 小端向大端的转换
使用移位。
对32位:
最高字节:右移24位
次高字节:右移8位
次低字节:左移8位
最低字节:左移24位
uint32_t reversebytes_uint32t(uint32_t value){
return (value & 0x000000FFU) << 24 | (value & 0x0000FF00U) << 8 |
(value & 0x00FF0000U) >> 8 | (value & 0xFF000000U) >> 24;
}
对64位:
先将高32位转小端模式
低32位转小端模式
再将高32位和低32位互换
// 先将64位的低32位转成小端模式,再将64位的高32位转成小端模式
// 在将原来的低32位放置到高32位,原来的高32位放置到低32位
uint64_t reversebytes_uint64t(uint64_t value){
uint32_t high_uint64 = uint64_t(reversebytes_uint32t(uint32_t(value))); // 低32位转成小端
uint64_t low_uint64 = (uint64_t)reversebytes_uint32t(uint32_t(value >> 32)); // 高32位转成小端
return (high_uint64 << 32) + low_uint64;
}