下面以指针的笔试题,来对指针有一个更加全面的理解和认识。
代码:下列的程序结果是多少?
- #include
- int main()
- {
- int a[5] = { 1, 2, 3, 4, 5 };
- int* ptr = (int*)(&a + 1);
- printf("%d,%d", *(a + 1), *(ptr - 1));
- return 0;
- }
运行结果:
内存解析图:
总结:
1.了解指针类型的作用。+1或者-1跳过多少字节。
2.了解数组名的意义
代码:下面的代码必须在x86的环境下运行。结果为多少?
- #include
- struct Test
- {
- int Num;
- char* pcName;
- short sDate;
- char cha[2];
- short sBa[4];
- }*p;
- //该结构体大小为20字节
- //假设p 的值为0x100000。
- int main()
- {
- printf("%p\n", p + 0x1);
- printf("%p\n", (unsigned long)p + 0x1);
- printf("%p\n", (unsigned int*)p + 0x1);
- return 0;
- }
运行结果:
代码分析:
- #include
- struct Test
- {
- int Num;
- char* pcName;
- short sDate;
- char cha[2];
- short sBa[4];
- }*p;
- //该结构体大小为20字节
- //假设p 的值为0x100000。
- int main()
- {
- printf("%p\n", p + 0x1);
- //0x1为十六进制的1,结构体指针+1,跳过一个结构体
-
- printf("%p\n", (unsigned long)p + 0x1);
- //结构体变量p被强转成了整数,整数+1,结果+1
-
- printf("%p\n", (unsigned int*)p + 0x1);
- //p被强转成了整形指针类型,+1跳过一个指针类型,就是4
-
- return 0;
- }
总结:
1.了解结构体内存大小的计算
2.不同的数据类型+1或者-1意义不一样
代码:
- #include
- 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;
- }
结果运行:
内存布局图:
总结:
1.数据的大小端存储
2.指针拿取数据的方式
代码:
- #include
- int main()
- {
- int a[3][2] = { (0, 1), (2, 3), (4, 5) };
- int* p;
- p = a[0];
- printf("%d", p[0]);
- return 0;
- }
运行结果:
本题误点:
int a[3][2]={(0,1),(2,3),(4,5)};这并不是直接给数组赋值,里面的而是逗号表达式
代码:
- #include
- 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;
- }
结果:
内存布局图:
打印方式解析:
总结:
1.找出两个地址的位置
2.计算出两个指针相减的值
3.两种不同的打印方式,所打印的值
代码:
- #include
- 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;
- }
内存分布图:
代码:
- #include
- int main()
- {
- char* a[] = { "work","at","alibaba" };
- char** pa = a;
- pa++;
- printf("%s\n", *pa);
- return 0;
- }
内存分布图:
代码:
- #include
- 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;
- }
结果展示:
数组指针分布图:
打印方式解析:
(1)
printf("%s\n", **++cpp);
(2)
printf("%s\n", *-- * ++cpp + 3);
cpp++或者cpp--会影响指针的指向,+1或者-1不会
(3)
printf("%s\n", *cpp[-2] + 3);
(4)
printf("%s\n", cpp[-1][-1] + 1);
总结:
熟练掌握指针与数组解引用操作的相互转换