概念
一级指针变量存放别的变量的地址,一级指针也有自己的地址,接下来的二级指针就是用来存放一级指针的地址,然而二级指针也有自己的地址,可以用三级指针来存放二级指针的地址,这样的方式像套娃。
为什么要用二级指针
就像一个在主函数的普通变量,通过传参,传到函数里想要进行修改数值,然后还给主函数,发现还是原来的数据。这时就需要把这个变量的地址给传上去进行修改了。二级指针也一样,主函数的一级指针要把自己的地址传上去,才能进行修改,主函数里也进行改变。
二级指针和二维数组容易犯的错误
如果直接用一个二级指针保存二维数组的首地址,如下图:
int data[3][2] = {{1,2}{3,4}{5,6}};
int **p2 = data;
printf("data的首地址和行的首地址=%p",data); //用数组名取地址
printf("data的首地址和行的首地址=%p",p2); //用二级指针取地址
printf("data的首地址和列的首地址=%p",*p2); //这里并不是表示列地址,这是一个野指针
int (*p)[2] = data;
printf("data的首地址和列的首地址=%p",*p); //这里表示列地址
printf("data的首地址和列的首地址=%p",*data); //这里表示列地址
如果碰到 *p2,它表示的是一个野指针,并不是像一级指针保存二维数组一样 *p就是指向数组的列地址。所以运用二级指针保存数组需要一级一级来保存,先用一级指针保存二维数组的首地址,再用二级指针保存一级指针的地址。
int data[3][2] = {{1,2}{3,4}{5,6}};
int (*p)[2] = data;
int **p2 = &p;
printf("data的首元素=%p",**p2);
为什么要用到二级指针
通过一个例子来说明:这里有三位学生的成绩,每个学生有4门成绩,输出0,1,2获得他的成绩。
#include <stdio.h>
void getstudent(int pos, int (*pstu)[4], int **p2) //这里用二级指针保存一级指针的地址
{
*p2 = (int *)(pstu+pos); //这里是偏移了输入了pos的位置,并指向这一行的首地址
//这里p2加了 * 意思是main函数里p的地址,这里是正在修改p的指向(地址)
//(int *)是为了强转为同一类型,都是整型指针变量
}
int main()
{
int students[3][4] = {{98,87,85,88},{92,96,87,99},{91,91,87,93}};
int pos,i,j;
int *p = students;
printf("请输入你要了解的同学的学号:(0,1,2)");
scanf("%d",&pos);
getstudent(pos,students,&p); //这里把指针的地址传上去,改动p的值,main函数里也跟着改动
for(i=0;i<4;i++)
{
printf("%d ",*p++); //输出的是一行,也就是一维数组
}
return 0;
}