大端和小端
大端:高位字节放在内存的低地址端,低位字节放在内存的高地址端。
小端:就是低位字节放在内存的低地址端,高位字节放在内存的高地址端。
简单的来记就是大端存储 字节和地址是相反的。
举一个例子,比如数字
0x12 34 56 78
在内存中的表示形式为:
- 大端模式:
低地址 —————–> 高地址
0x12 | 0x34 | 0x56 | 0x78
- 小端模式:
低地址 ——————> 高地址
0x78 | 0x56 | 0x34 | 0x12
判断大端小端
#include<stdio.h>
int main(){
int a = 0x11223344;
char *p;
p = (char*)&a;
printf("%x", *p);
return 0;
}
定义整型变量a,指针p指向a的首地址,也就是低地址,输出p查看,若输出11,说明高位存储在了低地址,字节和地址是相反的,是大端存储;若输出44,说明低位存储在了低地址,是小端存储。
主机字节序和网络字节序
现代PC大多采用小端字节序,因此小端字节序又被称为主机字节序。
当格式化的数据(比如32bit整型数和16bit短整型数)在两台使用不同字节序的主机之间直接传递时,接收端必然会错误的解释之。解决的办法是:发送端总是把要发送的数据转化成大端字节序数据后再发送,而接收端知道对方传送过来的数据总是采用大端字节序,所以接收端可以根据自身采用的字节序决定是否对接受到的数据进行转换(小端机转化,大端机不转化)。因此大端字节序也成为网络字节序,它给所有接收数据的主机提供了一个正常解释收到的格式化数据的保证。
主机字节序和网络字节序转换:
linux提供如下4个函数来完成主机字节序和网络字节序之间的转换。
#include<netinet./in.h>
unsigned long int htonl(unsigned long int hostlong);
unsigned short int htons(unsigned short int hostshort);
unsigned long int ntohl(unsigned long int netlong);
unsigned short int ntohs(unsigned short int netshort);
比如htonl表示“host to network long”,将长整型的主机字节序数据转化为网络字节序数据。
这四个函数中,长整型函数通常用来转换IP地址,短整型函数用来转换端口号。
(当然不限于此,任何格式化的数据通过网络传输时,都应该使用这些函数来转换字节序)。