目录
一、二级指针的介绍
首先指针是存放变量的地址的,通过地址访问变量的值。而二级指针 int **p 存放的是指针变量的地址。下面我们用一级指针的不足来引入二级指针。
1.测试代码
int data = 10;
int* p = &data;
printf("data的地址:%p\n", &data);
printf("指针p存放的data的地址:%p,其内容为%d\n", p, *p);int* p1 = &p;
printf("p1所保存的地址是%p\n", p1);
printf("*p1所保存的地址是%p\n", *p1);
2. 测试的结果:
我们把它用图形的形式抽象出来,如下图所示
由上边我们可以看到指针*p1是无法访问data的值的 ,不能多级进行访问,这个时候就需要引入一个二级指针。
3.二级指针的引入
int** p2 = &p; //p2保存的是指针变量p的地址,每一个变量都有它自己的地址
printf("p2所保存的地址是%p\n", p2); //就是p的地址
printf("*p2所保存是%p\n", *p2); //保存的是data的地址,也就是p所指向的data的地址
printf("**p2访问data的值是:%d\n", **p2); //这里就取得了data的值:10
以下是运行结果
(这里重新运行了,地址重新分配了,没有影响)
我们把它用图形的形式抽象出来,如下图所示
同理还有三级指针,不过实际不咋用。
二、二级指针的应用
这里我用学习过程的一个题目为例,3个学生4门科目,输入指定的学生号数,返回他相应的科目,代码里面注释的是用指针函数写的,通过返回对应数组的行地址,来传回主函数的ppos指针。下面的是输出里面指定成绩不合格(60以下)的同学位号。
#include <stdio.h>
/*
* 指针一般考虑起始地址,偏移量。
* 这里考虑起始地址
*/
//int* getSnoPerson(int sno, int(*pstu)[4]) //指针函数
//{
// int* p;
// p = pstu + sno - 1;
// return p;
//}
//假如返回值是空
void getSnoPerson(int sno, int(*pstu)[4], int** ppos) {
*ppos = pstu + sno - 1;
}
int main() {
int arr[3][4] = {
{11,22,33,44},
{22,23,33,42},
{61,62,63,66}
};
int* ppos;
int sno;
printf("请输入你想看的学生学号:(1,2,3):");
scanf("%d", &sno);
getSnoPerson(sno,arr,&ppos);
for (int i = 0; i < 4; i++) {
printf("%d\t", *ppos++);
}
/*for (int i = 0; i < 3; i++)
{
for (int j = 0; j < 4; j++) {
if (arr[i][j] < 60)
{
sno = i;
printf("sno:%d\t", sno + 1);
i++;
continue;
}
}
}*/
}
1.解析
这里我们返回值为空,在进入到getSnoPerson函数时,系统会默认给函数的每个形参分配空间,在函数执行完毕,空间会被释放(也就是临时空间),我们需要通过传递ppos指针的地址,然后通过二级指针ppos,*ppos就是ppos(这里指main函数里面的ppos指针)所存储的地址(也就是通过操作二级指针里的ppos来间接操作主函数里面的ppos,让它指向二维数组相应的行地址),最后再回到主函数,输出它所有的科目成绩.
tips:
getSonPerson函数的第二个参数是数组指针(行指针)——提醒一下
三、二级指针与二维数组的关系
当二级指针指向二维数组名时,二级指针p表示里面的数据也就是二维数组的首地址,但是*p不存在(发生段错误——对野指针或者空指针进行赋值)
总结,所谓的关系也就是二级指针与行指针(也就是数组指针 int (*p)[4])。
四、总结
以上均为个人学习总结,为个人观点,仅供参考。
二级指针首先要清楚指针类型的数据是地址(p是普通变量的地址),而二级指针指向的是指针变量的地址(换句话来说,就是取得内容是指针变量的内容也就是地址)