前言
紧跟上一章,这一章的指针相对较难一些,不过经过本章的学习,相信大家能够有所得
一.二级指针
- 上一章说的都是一级指针的应用,那么什么是二级指针呢?
我们都知道,创建一个变量是需要在内存中开辟一个地址空间来存放的,那么指针(指针就是指针变量)也是如此,他也是一个变量,他也需要一个地址内存空间来存放,只不过这个空间放的是其他变量的地址。
实际上,二级指针可以理解为就是指针的指针,也就是说,二级指针里放的是一级指针的地址。
我们创建一个整型变量给个初始值,再将这个变量的地址交给一个指针变量,然后将这个指针的地址再传给另一个指针,那么这个指针就叫做二级指针。
我们看以下代码:
int a = 10; // 创建一个变量a,他需要在内存中占有一段空间 int* pa = &a; // pa为指针变量,他也需要一段空间来存放,但他里面放的是a的地址 int** ppa = &pa; // 同理,它里面放的是pa这个指针的地址
》》 那么我们如何使用二级指针呢?
- 由上述代码,我们都知道,对一级指针解引用操作可以找到a的值,并且能够修改a里面的值,同样的。对ppa解引用就能找到a的地址,因为指针变量pa存放的是a的地址,如下:
#include <stdio.h> int main() { int a = 10; int* pa = &a; int** ppa = &pa; printf("%p\n", &a); printf("%p\n", pa); printf("%d\n", *pa); printf("%p\n", &pa); printf("%p\n", ppa); // ppa就是pa的地址 printf("%p\n", *ppa); // 对ppa解引用找到的是a的地址,因为pa里面存放的是a的地址 return 0; }
- 也就是说通过对ppa两次的解引用,我们可以找到a的值:
- ppa的解引用过程:一次解引用找到的是a的地址,因为ppa是指向int 类型的指针,他指向int型的指针变量pa,而pa里面存放的是a的地址。二次解引用便通过a的地址找到a的值。
二.指针数组
- 指针数组他是数组还是指针呢???
- 当然是数组。
- 数组里的元素放的是指针,那这个数组就是指针数组。
例如:
int* arr[3] = {0};
arr是一个指针数组,他里面放了三个指针,每个指针是指向int类型的指针。
那么我们如何使用二维数组,下面展示一个比较能够体现其作用的代码:
#include <stdio.h> int main() { int a[4] = { 1,2,3,4 }; int b[4] = { 5,6,7,8 }; int c[4] = { 9,10,11,12 }; int* arr[3] = { a,b,c }; int i = 0; int j = 0; for (i = 0; i < 3; i++) { for (j = 0; j < 4; j++) { printf("%-2d ", *(arr[i] + j)); // arr[i][j] == *(arr[i] + j) } printf("\n"); } return 0; }
可以看到,指针数组arr里面有三个指针,他们分别指向数组a的首元素地址,数组b的首元素地址,数组c的首元素地址,后面我们相继打印每个数组的内容,我们利用 *(arr[i] + j) 来寻找到每个数组对应下标的内容,arr[i]表示找到指针数组对应下标的指针指向的首元素地址,+j表示指针偏移找到该数组后面的元素的地址,对整体解引用则为该指针对应的数组的对应下标的值。
总结
指针初阶就到这里,想必大家对指针的认识已有不少。我们都知道,灵活的利用指针可以大大提高程序的效率,所以这里我们需要针对性的练习来提高我们对指针的敏感度。