题目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]); //1 return 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);// at return 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"这种操作是有"副作用"的运算,它会改变本身的内容!