指针进阶(详解)
创始人
2024-05-08 13:00:25
0

指针进阶

  • 一.字符指针
  • 二.指针数组
    • 1.一次打印多个字符串
    • 2.模拟二维数组
  • 三.数组指针
    • 1.定义
    • 2.应用
  • 四.函数指针
  • 五.函数指针数组
  • 六.指向函数指针数组的指针
  • 七.回调函数

在这里插入图片描述

在开始这篇之前,前面有两篇指针初阶,如果需要的话可以去看看哟!指针初阶1,指针初阶2

一.字符指针

在这里插入图片描述

对于一个字符串,那么指针又该如何指向呢?

在这里插入图片描述

毫无疑问p里存的是地址,那它存了abcdefg的所有地址吗?当然不是,因为它根本存不下(一个指针在32为机器下只有4个字节,如果不太明白可以看看指针初阶)。实际上,它存的是首字符a的地址。字符串在内存里是连续存放的,所以只需要找到首元素地址就可以啦。ps:这和数组很像,事实上处理字符串时也可以按照下标来处理。

在这里插入图片描述

追加一个小知识:字符串可以直接使用%s打印是因为它有\0作为结束标志。但如果全部是整数的话,没有结束标志就只能通过下标挨个打印。

一道面试题

在这里插入图片描述

在这里插入图片描述

首先,str1和str2是两个数组,需要开辟两个空间,而数组名代表首元素地址,空间不同首元素地址自然不同,故str1!=str2。但是str3和str4是指针变量,里面存的都是h的地址并且hello,bite是一个常量字符串永远不能被改变。所以此时编译器就会认为既然都一样且没法被改变,那么就只会开辟一个空间,str3和str4都指向同一空间,故str3==str4。(可以理解为编译器的一种规定)

二.指针数组

存放整形的数组->整形数组,存放字符的数组->字符数组,存放指针的数组->指针数组。

1.一次打印多个字符串

在这里插入图片描述

在这里插入图片描述

我们如何打印呢?很简单,因为数组内每个元素存的都是各个字符串首元素地址,所以我们只需要依靠首元素地址就能打印出所有字符串。

在这里插入图片描述
在这里插入图片描述

2.模拟二维数组

在这里插入图片描述

它的打印需要使用两个for嵌套,因为它不能像字符串那样打印一串,这里其实也就是用一维数组模拟二维数组(但不同的是二维数组每一行每一列都是连续存放的,而这里每一行不一定连续)

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

三.数组指针

1.定义

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

追加个小知识:【】的优先级高于*所以别忘了加括号哦。

ps:如果pa先与*结合就说明是个指针,如果先于【】结合就说明是个数组。

pps:这里的&arr是整个数组的地址,也就是如果+1的话是跳过整个数组。注意与数组名区分开。

2.应用

1.强行使用

在这里插入图片描述

这样写实际上很别扭,当然实际上一般也没人会怎么写。所以它的真正作用体现在二维数组里。

2.正常使用

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

一些练习

在这里插入图片描述

四.函数指针

指向函数的指针。

一段代码

首先,函数名=&函数,故函数名就是函数地址(如果需要验证的话,写一个函数再用%p输出就可以了,这里就不再累述了)

在这里插入图片描述

分析这个指针的构成,首先是一个指针故pf应该先与*结合,其次指向一个函数,后面就应该接上(参数),最后该函数是int类型,故再前面写上int。将这3个部分串起来就是如上指针。

接下来使用pf指针调用Add函数。

在这里插入图片描述

在这里插入图片描述

首先对pf进行解引用,解引用后就是原函数。也就是*pf=Add,接下来是传参(参数1,参数2),这样就完成了调用功能。实际上,在调用时前面的星号是没有作用的,因为如我们平常调用函数Add(2,3),pf里存的本身就是Add,所以不需要解引用。

在这里插入图片描述

代码一

在这里插入图片描述

这道题的关键就是把0当成地址来看。

在这里插入图片描述

该代码是一次函数调用,调用0地址处的函数

ps:以上代码出自c语言缺陷与陷阱

代码二

在这里插入图片描述

依然从名字signal入手。

在这里插入图片描述

1.该代码是一次函数的声明。
2.声明的函数名字是signal
3.参数有两个,第一个是int类型,第二个是函数指针类型
4.signal函数的返回值是一个函数指针

五.函数指针数组

数组是一个存放相同类型数据的存储空间,那我们已经学习了指针数组,比如:int arr[10];数组的每个元素是int。那要把函数的地址存到一个数组中,那这个数组就叫函数指针数组,那函数指针的数组如何定义呢?

在这里插入图片描述

使用:简单的计算器

int add(int a, int b){return a + b;
}
int sub(int a, int b){return a - b;
}
int mul(int a, int b){return a * b;
}
int div(int a, int b){return a / b;
}
int main()
{int x, y;int input = 1;int ret = 0;do{printf("*************************\n");printf("*** 1:加法      2:减法***\n");printf("*** 3:乘法      4:除法*** \n");printf("**********0.退出*********\n");printf("*************************\n");printf("请选择:");scanf("%d", &input);switch (input){case 1:printf("输入操作数:");scanf("%d %d", &x, &y);ret = add(x, y);printf("ret = %d\n", ret);break;case 2:printf("输入操作数:");scanf("%d %d", &x, &y);ret = sub(x, y);printf("ret = %d\n", ret);break;case 3:printf("输入操作数:");scanf("%d %d", &x, &y);ret = mul(x, y);printf("ret = %d\n", ret);break;case 4:printf("输入操作数:");scanf("%d %d", &x, &y);ret = div(x, y);printf("ret = %d\n", ret);break;case 0:printf("退出程序\n");break;default:printf("选择错误\n");break;}} while (input);return 0;
}

这个代码很简单就不多说了。实际上我们可以看到这个代码很冗长繁琐,如果在之后我们需要给计算器添加新的功能的话,毫无疑问case会加长,我们能不能有一种方法避免这种写法呢?

int add(int a, int b){return a + b;
}
int sub(int a, int b){return a - b;
}
int mul(int a, int b){return a * b;
}
int div(int a, int b){return a / b;
}
int main()
{int x, y;int input = 1;int ret = 0;int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //使用数组去除冗长的调用while (input){printf("*************************\n");printf("*** 1:加法      2:减法***\n");printf("*** 3:乘法      4:除法*** \n");printf("**********0.退出*********\n");printf("*************************\n");printf("请选择:");scanf("%d", &input);if (0 == input){printf("退出程序。\n");break;}if ((input <= 4 && input >= 1)){printf("输入操作数:");scanf("%d %d", &x, &y);ret = (*p[input])(x, y);}elseprintf("输入有误\n");printf("ret = %d\n", ret);}return 0;
} 

使用函数指针数组的前提是所有的函数类型都相同,都是两个参,参数类型都是int,返回值都是int。

六.指向函数指针数组的指针

其实本质上就是一个数组指针,只不过这个数组的类型是一个函数指针类型数组。

在这里插入图片描述

要分辨类型,需要先看与哪个操作符结合。

一些练习

在这里插入图片描述

七.回调函数

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个 函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

应用例子qsort函数,由于篇幅比较大,将其放到了下一篇博客里qsort函数

在这里插入图片描述

相关内容

热门资讯

监控摄像头接入GB28181平... 流程简介将监控摄像头的视频在网站和APP中直播,要解决的几个问题是:1&...
Windows10添加群晖磁盘... 在使用群晖NAS时,我们需要通过本地映射的方式把NAS映射成本地的一块磁盘使用。 通过...
protocol buffer... 目录 目录 什么是protocol buffer 1.protobuf 1.1安装  1.2使用...
在Word、WPS中插入AxM... 引言 我最近需要写一些文章,在排版时发现AxMath插入的公式竟然会导致行间距异常&#...
【PdgCntEditor】解... 一、问题背景 大部分的图书对应的PDF,目录中的页码并非PDF中直接索引的页码...
Fluent中创建监测点 1 概述某些仿真问题,需要创建监测点,用于获取空间定点的数据࿰...
educoder数据结构与算法...                                                   ...
MySQL下载和安装(Wind... 前言:刚换了一台电脑,里面所有东西都需要重新配置,习惯了所...
修复 爱普生 EPSON L4... L4151 L4153 L4156 L4158 L4163 L4165 L4166 L4168 L4...
MFC文件操作  MFC提供了一个文件操作的基类CFile,这个类提供了一个没有缓存的二进制格式的磁盘...