不同的CPU有不同的字节序类型这些字节序是指整数在内存中保存的顺序这个叫做主机序
最常见的有两种:
1. Little endian:将低序字节存储在起始地址
2. Big endian:将高序字节存储在起始地址
LE little-endian
最符合人的思维的字节序
地址低位存储值的低位
地址高位存储值的高位
怎么讲是最符合人的思维的字节序,是因为从人的第一观感来说
低位值小,就应该放在内存地址小的地方,也即内存地址低位
反之,高位值就应该放在内存地址大的地方,也即内存地址高位
BE big-endian
最直观的字节序
地址低位存储值的高位
地址高位存储值的低位
为什么说直观,不要考虑对应关系
只需要把内存地址从左到右按照由低到高的顺序写出
把值按照通常的高位到低位的顺序写出
两者对照,一个字节一个字节的填充进去
例子:在内存中双字0x01020304(DWORD)的存储方式
内存地址
4000 4001 4002 4003
LE 04 03 02 01
BE 01 02 03 04
例子:如果我们将0x1234abcd写入到以0x0000开始的内存中,则结果为
big-endian little-endian
0x0000 0x12 0xcd
0x0001 0x34 0xab
0x0002 0xab 0x34
0x0003 0xcd 0x12
x86系列CPU都是little-endian的字节序.
网络字节顺序是TCP/IP中规定好的一种数据表示格式,它与具体的CPU类型、操作系统等无关,从而可以保证数据在不同主机之间传输时能够被正确解释。网络字节顺序采用big endian排序方式。
下面我们以IP:127.0.0.1为例说明主机字节与网络字节的转换:
第一步 : 127 . 0 . 0 . 1 把IP地址每一个字节转换为8位的二进制数。
第二步: 01111111 00000000 00000000 00000001 = 2130706433 (主机字节序) (2130706433为对应的十进制数值)
然后把上面的四部分二进制数从右往左按部分重新排列,那就变为:
第三步: 00000001 00000000 00000000 01111111 = 16777343 (网络字节序)(16777343 为对应的十进制数值)
同理,网络字节转换成主机字节正好相反。
虽然网络编程中有htons(),htonl(),ntohs()等接口函数,但是亲自实现还是很有意思的哈,
编程实现:
//主机字节转换成网络字节的端口
u_short hostToNets( u_short hostPort)
{
u_short netPort=0;
netPort= (hostPort&127)<<8;
netPort +=(hostPort & 65280)>>8;
return netPort;
}
//网络字节转换成主机字节的端口
u_short netToHosts( u_short netPort)
{
u_short hostPort=0;
hostPort=(netPort & 127)<<8;
hostPort +=(netPort & 65280)>>8;
return hostPort;
}
//IP 整数小端向大端转换
u_long hostToNetl(u_long hostIp)
{
u_long netIp=0;
netIp= (hostIp&127)<<24;
netIp +=( hostIp & 65280)<<8;
netIp +=( hostIp & 16711680)>>8;
netIp +=( hostIp & 4278190080 )>>24;
return netIp;
}
//IP整数大端向小端转换
u_long netToHostl(u_long netIp)
{
u_long hostIp=0;
hostIp= (netIp&127)<<24;
hostIp +=( netIp & 65280)<<8;
hostIp +=( netIp & 16711680)>>8;
hostIp +=( netIp & 4278190080 )>>24;
return hostIp;
}