一、认识大小端
1、大小端是什么?
大小端是机器内存的存储方式,分为大端存储和小段存储。
2、具体大小端是怎么存储的?
大端:低位地址存储在内存高位 小段:低位地址存储在内存低位
图示:
3、为什么要需要大小端?
因为数据会乱套,不同计算机的存储方式不一样。如果两台计算机需要进行通信,如果存储方式不一样。传输数据一个是小端传输,一个是大端接收。小端将内存中的数据“abcd”发送,大端接收并放在内存数据是“dcba”,数据就乱套了。
如果提起发送端提前知道接收数据的大小端,或者约定好大小端,那么数据就不会出错。例如:小端内存中原数据是“abcd”,但是知道接收端是大端的模式,小端将发送的数据改为“dcba”,大端就能准确接收到原数据“abcd”到内存中了。
二、大小端使用
1.大小端判定
大小端判定是用内存的存储方式的特点进行判断。判断方法一般有两种,一种是联合体法(需要知道联合体的存储特点),一种是利用指针的方法。其核心都是读取多于两位的数据的低位,查看低位数据数据是否存放在低位。
union Dx
{
int a;
char c;
}un;
int judgeDx()
{
un.a=1;
if(un.c==1) return 0;
else return 1;
}
int judgePx()
{
int a=1;
char *c=(char *)&a;
if(*c==1) return 0;
else return 1;
}
2.大小端转换
大小端转换也是利用内存的存储特点进行转换的,其核心也就是交换位置。将高地址内存放到低地址内存,或者将低地址内存放到高地址内存中。
大小端转换一般会利用移位操作,或者其他内存交换操作。我这里写了一个效率不算高的,但是通用性好的大小端转换的函数。移位操作下面也可以提高学习。
#define SWP16(X) (( ((uint16)(X) & 0xff00) >> 8) | \
(( (uint16)(X) & 0x00ff) << 8))
#define SWP32(X) ((( (uint32)(X) & 0xff000000) >> 24) | \
(( (uint32)(X) & 0x00ff0000) >> 8) | \
(( (uint32)(X) & 0x0000ff00) << 8) | \
(( (uint32)(X) & 0x000000ff) << 24))
void LargeSmallConver(void *data,int size)
{
unsigned char tmp;
unsigned char *tdata=(unsigned char *)data;
int hight=size-1,low=0;
while(low<hight)
{
tmp=tdata[low];
tdata[low]=tdata[hight];
tdata[hight]=tmp;
low++;
hight--;
}
}
int main()
{
int i;
char test[4]={'a','b','c','d'};
printf("before %s\r\n",test);
LargeSmallConver(test,sizeof(test));
printf("after %s\r\n",test);
system("pause");
}
双指针法图解:
|