怎么去理解大端和小端?
1、大端和小端核心是什么?
大端模式和小端是实际的字节顺序和存储的地址顺序对应关系的两种模式,总结如下:
大端模式:低地址对应高字节
小端模式:低地址对应低字节
不管是大端还是小端模式,我们在读取和存储数据的时候一定都是从内存的低地址依次向高地址读取或写入。
2、什么是存储顺序?
打个比方,我们定义一个数组,char array[5] = {0,1,2,3,4};
学习过C语言应该都知道,array是这个数组存储的首地址,如果这个您也不知道,那我还是建议你去复习一下C语言,这是C语言最基本不过的内容了。
数组存储的内存分布如下:
存储地址 |
0x2000 3001 |
0x2000 3002 |
0x2000 3003 |
0x2000 3004 |
0x2000 3005 |
存储的值 |
0 |
1 |
2 |
3 |
4 |
不管是写还是读,我们都是只要根据这个首地址就能找到我们想要的元素内容,从这一点上讲,在存储的时候,我们依次去存储0、1、2、3、4。在读取的时候,我们也是从array[0]开始,依次读取0、1、2、3、4。
3、大端模式向小端模式发送数据
有了数据按照字节依次由低往高读取或存储这个前提,下面举例说明大端和小端的区别。
假设A要发送四字节的数据给B,A的存储是按照大端模式,B的存储模式为小端模式。
A的存储1234内存分布如下图
存储地址 |
0x2000 4001 |
0x2000 4002 |
0x2000 4003 |
0x2000 4004 |
存储的值 |
0x12 |
0x34 |
0x56 |
0x78 |
因为A存储是按照大端模式,依据低地址对应高字节的规律,那么A要发送值int Value_A = 0x12345678。
Value_A将会由低地址依次发送给B,这里为什么是低地址,因为发送的开始,就是去读取数据,读取数据一定是从低地址开始读取得,那么B依次接收的值为0x12,0x34,0x560,x78。
B将接收到的值存储在如下图的内存中,存储的时候也是按照低地址往高地址开始依次存储
存储地址 |
0x2000 8001 |
0x2000 8002 |
0x2000 8003 |
0x2000 8004 |
存储的值 |
0x12 |
0x34 |
0x56 |
0x78 |
B接收完毕后,需要将内存中的值读取到Value_B中来,那么Value_B的值是多少呢?
B的存储是按照小端模式存储,低地址对应的低字节,那么int Value_B = 0x78563412
这里可能很多人会有疑问?不是读取是按照低地址从高地址依次读取的吗?为什么相同的存储,读出来的值却不一样了?
这里就是关键了。在读取0x12出来后,系统设定的模式是低地址对应低字节,我们的Value_B
是int类型,是四个字节的数据,展示在我们面前的数据,一定是从左到右字节的顺序依次降低的,所以将0x12放在最低的字节,如下图。
字节顺序 |
4 |
3 |
2 |
1 |
对应的字节的值 |
0x78 |
0x56 |
0x34 |
0x12 |
即Value_B = 0x78563412;
Value_A 不等于Value_B,这就是为什么我们在网络通信的时候一定要进行字节序和确认了,必须保证A和B的字节序相同,如果不同,就需要使用字节序的转换函数。
4、字节序转换函数
1、htons 把unsigned short类型从主机序转换到网络序(host to network short)
2、htonl 把unsigned long类型从主机序转换到网络序(host to network long)
3、ntohs 把unsigned short类型从网络序转换到主机序(network to host short)
4、ntohl 把unsigned long类型从网络序转换到主机序(network to host long)
主机字节序一般都是小端(绝大多数,少部分也是大端存储的),网络字节序是大端存储的。
5、小端转大端函数讲解和原理分析
下面是一个将小端转化为大端的函数:
UINT32 LE2BE(UINT8 *dat,UINT8 len)
{
UINT32 temp=0,fact=1;
UINT8 i=0;
for(i=0;i { temp+=dat[i]*fact; fact*=256; } return temp; } 这段程序是什么意思呢? 我们还是举例说明。 在上面的案例中,如果B要给A发送int test_B = 0x12345678,B的存储就是小端模式,B要将0x12345678正确的发送给A,那么按照上面的发送和接收,A是不会接收到0x12345678的。 A会依次接收到0x78,0x56,0x34,0x12,不经过转换,低地址对应高字节,test_A = 0x78563412。 这里就需要按照上面进行转换。 上面的程序意图如下: temp = 0x78*1 + 0x56*256 + 0x34 * 256*256 + 0x12*256*256*256 = 120+22016+3407872+301989888 = 305419896 = 0x12345678 经过转换,A获取的值就是temp值,也就是B传送过来的值了。 看到这里,就算是把大端和小端的问题说明白了,实际上的问题核心就是一个是字节顺序,一个存储顺序。