地址的强制转换和其转换的类型有很大的关系
下面来看《C语言深度解剖》上面的一道很有意思的题。
int main()
{
int a[4]={1,2,3,4};
int *ptr1=(int*)(&a+1);
int *ptr2=(int*)((int)a+1);
printf("%x,%x",ptr1[-1],*ptr2);
return 0;
}
问:在X86系统下,其输出为多少?
1.&a+1和a+1分别代表什么意思?
我们知道a是数组首元素的地址,而&a代表的是取出整个数组的地址,虽然其值一样但是代表的意义是不同的。除了sizeof(数组名)和&(数组名)之外,所有的数组名都表示数组首元素的地址。
2.强制转换
上面我们知道了ptr1=&a+1其实就等于ptr1=&a+sizeof(int)*4;而ptr2=(int)a+1=a+(int)1;
假设起始地址为0x00000000。ptr2中a被强制转化为int型,(int)a+1就等于int(0x00000000)+1 = 0+1。一个地址搭配一个字节。
3.机器的大小端问题。
我们由上图知道了数组a在内存中的排列和ptr1、ptr2指向哪里。但是要知道这个代码的结果还得看你的机器到底是大端还是小端了。这可真是个鸡儿问题,很多人在这里都跳到坑里面了。
小端无非就是低地址存低字节,高地址存高字节;大端是低地址存高字节,高地址存低字节内存中地址是线性排列的,并且是由低到高排列的。
4.求大小端的两种方法
(1)利用指针的特性
int sys_check()
{
int i=1;
char *c=&i;
return (*c);
}
int main()
{
int ret=sys_check();
if(ret==1)
{
printf("小端");
}
else
{
printf("大端");
}
}
指针的类型是多少字节的,则在其解引用时就会解引用多少字节的数据。(这句话不太严谨)
(2)利用共用体(联合体)
int sys_check()
{
union check
{
int i;
char ch;
}c;
c.i=1;
return (c.ch==1) //如果ch里面是1的话,那么c.ch==1成立,也就是return 1
}
int main()
{
int ret=sys_check();
if(ret==1)
{
printf("小端");
}
else
{
printf("大端");
}
}
利用共用体变量公用同一块内存的特性。
好了,终于所有的分析工作都做完了。那么当系统是小端的时候,*ptr2=0x02000000,ptr1[-1]=0x00000010;大端时:*ptr2=0x00000100,ptr1[-1]=0x00000010。