题目1:
#include <stdio.h>int main()
{int a[5] = { 1, 2, 3, 4, 5 };int* ptr = (int*)(&a + 1);printf("%d,%d", *(a + 1), *(ptr - 1));return 0;
}
//程序的输出结果是什么? 2 5
解析:
代码的内存布局如图:
- 考点:
- 理解数组名与&数组名的区别。
数组名一般是指数组首元素地址(除sizeof(数组名)外);&数组名取出的是整个数组的地址。两者的值相同,意义不一样。- 理解指针±整数操作。
指针±整数的步幅是多少,取决于指针的类型,如整型指针一次解引用访问4个字节,字符指针一次解引用访问1个字节。
题目2:
#include <stdio.h>//在X86环境下
//假设结构体的⼤⼩是20个字节
//程序输出的结果是啥? struct Test
{int Num;char* pcName;short sDate;char cha[2];short sBa[4];
}* p = (struct Test*)0x100000;int main()
{printf("%p\n", p + 0x1);printf("%p\n", (unsigned long)p + 0x1);printf("%p\n", (unsigned int*)p + 0x1);return 0;
}
解析:
代码解析如图:
所以,第一个printf语句中,p+0x1就等价于指针±整数操作,步幅取决于指针类型的大小,由题可知结构体大小为20个字节,所以结果为0x100000+14(20要转换为十六进制)=0x100014。
第二个printf语句中,p是无符号整型,就是整数,整数+1,结果就是0x100001。
第三个printf语句中,p是整型指针,+1跳过4个字节,结果为0x100004。- 考点:指针±整数。指针±整数的步幅是多少,取决于指针的类型。
题目3:
#include <stdio.h>int main()
{int a[3][2] = { (0, 1), (2, 3), (4, 5) };int* p;p = a[0];printf("%d", p[0]); //1return 0;
}
解析:
代码内存布局如图:
这道题也是考查对数组名的理解,a[0]是二维数组第一行的数组名,而数组名又是数组首元素地址,所以p[0]=1。
- 考点:对二维数组的数组名的理解。
题目4:
//假设环境是x86环境,程序输出的结果是啥? #include <stdio.h>int main()
{int a[5][5];int(*p)[4];p = a;printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);return 0;
}
解析:
代码的解析和内存布局如图:
|指针-指针| = 两者之间元素的个数。由于数组在内存中是由低地址向高地址存储的,由图可知,两个地址的差是-4,%d打印有符号整型,结果为-4;但是%p是以十六进制打印补码,-4的原码为10000000000000000000000000000100,所以补码为11111111111111111111111111111100,转换为十六进制为FFFFFFFC。
题目5:
#include <stdio.h>int main()
{int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };int *ptr1 = (int *)(&aa + 1);int *ptr2 = (int *)(*(aa + 1));printf( "%d,%d", *(ptr1 - 1), *(ptr2 - 1));return 0;
}
解析:
代码内存布局如图:
&aa取出的是整个二维数组的地址,+1跳过整个二维数组;而aa要理解为二维数组第一行的地址,+1跳过一整行。
题目6:
#include <stdio.h>int main()
{char *a[] = {"work","at","alibaba"};char**pa = a;pa++;printf("%s\n", *pa);// atreturn 0;
}
解析:
代码的内存布局如图:
这题主要考察对指针数组的理解和二级指针的理解。
指针数组是数组,里面的元素是地址,二级指针存放的是一级指针的地址。
题目7:
int main()
{char *c[] = {"ENTER","NEW","POINT","FIRST"};char**cp[] = {c+3,c+2,c+1,c};char***cpp = cp;printf("%s\n", **++cpp);printf("%s\n", *--*++cpp+3);printf("%s\n", *cpp[-2]+3);printf("%s\n", cpp[-1][-1]+1);return 0;
}
解析:
代码的内存布局如图:
这个题较为复杂,一定要正确的画出内存布局。并且一定要注意"++cpp和–cpp"这种操作是有"副作用"的运算,它会改变本身的内容!