C语言进阶:进阶指针(下)

一、 函数指针数组

我们都知道 数组是一个存放相同类型数据的存储空间 那我们已经学习了指针数组

那么函数有没有对应的指针数组呢? 如果有那应该怎么定义呢?

1. 函数指针数组的定义

我们说 函数指针数组的定义 应该遵循以下格式

int (*p[10])();

首先p和[]结合说明了它是一个数组
我们再将它拿掉可以发现 数组的内容就是
int (*)() 类型的函数指针。

2. 函数指针数组的用途:转移表(以计算器为例)

我们首先来写一段代码实现计算器的功能

void mune()
{printf("****** 1.add   2.sub *****\n");printf("****** 3.mul   4.div *****\n");printf("******    0.exit     *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{mune();printf("请选择:>");scanf("%d",&input);switch (input){case 1:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;case 2:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Sub(x, y);printf("%d\n", ret);break;case 3:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Mul(x, y);printf("%d\n", ret);break;case 4:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Div(x, y);printf("%d\n", ret);break;case 0:printf("退出计算器\n");break;default:printf("选择错误,请重新选择\n");}} while (input);return 0;
}

那么我们再使用函数指针来实现试试

void mune()
{printf("****** 1.add   2.sub *****\n");printf("****** 3.mul   4.div *****\n");printf("******    0.exit     *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}
//函数指针数组存放上述函数
//转移表
int (*pf[5])(int x, int y) = { NULL, Add, Sub, Mul, Div };int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{mune();printf("请选择:>");scanf("%d", &input);if (input == 0){printf("退出计算器\n");}else if (input >= 1 && input <= 4){printf("请输入两个操作符:>");scanf("%d %d", &x, &y);ret = pf[input](x, y);printf("%d\n", ret);}else{printf("选择错误\n");}} while (input);return 0;
}

二、回调函数

我们还是以计算器为例来解释回调函数的使用

void mune()
{printf("****** 1.add   2.sub *****\n");printf("****** 3.mul   4.div *****\n");printf("******    0.exit     *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{mune();printf("请选择:>");scanf("%d",&input);switch (input){case 1:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Add(x, y);printf("%d\n", ret);break;case 2:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Sub(x, y);printf("%d\n", ret);break;case 3:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Mul(x, y);printf("%d\n", ret);break;case 4:printf("请输入两个整数\n");scanf("%d %d", &x, &y);ret = Div(x, y);printf("%d\n", ret);break;case 0:printf("退出计算器\n");break;default:printf("选择错误,请重新选择\n");}} while (input);return 0;
}

我们可以发现 case1 2 3 4里面有大量冗余的代码

那么我们可不可以通过一定的手段解决这个问题呢?

答案是可以 就是使用回调函数

定义

回调函数就是一个被作为参数传递的函数。

我们来设计一个case_()函数 并且将add div这些函数作为参数传递进去

要求: 可以实现和原函数一样的功能 代码不能冗余

void calc(int (*pf)(int x, int y))
{int x = 0;int y = 0;int ret = 0;printf("请输入两个操作符:>\n");scanf("%d %d", &x, &y);ret = pf(x, y);printf("%d\n", ret);
}

这就是上面思想指导写出来的代码

完整代码如下

void mune()
{printf("****** 1.add   2.sub *****\n");printf("****** 3.mul   4.div *****\n");printf("******    0.exit     *****\n");printf("**************************\n");
}
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int Mul(int x, int y)
{return x * y;
}
int Div(int x, int y)
{return x / y;
}
void calc(int (*pf)(int x, int y))
{int x = 0;int y = 0;int ret = 0;printf("请输入两个操作符:>\n");scanf("%d %d", &x, &y);ret = pf(x, y);printf("%d\n", ret);
}int main()
{int input = 0;do{mune();printf("请选择:>");scanf("%d",&input);switch (input){case 1:calc(Add);break;case 2:calc(Sub);break;case 3:calc(Mul);break;case 4:calc(Div);break;case 0:printf("退出计算器\n");break;default:printf("选择错误,请重新选择\n");}} while (input);return 0;
}

让我们试试它的运行结果

可以完美运行的

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/650801.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

thinkphp5.0.23漏洞复现以及脚本编写

1 thinkphp5.0.23远程代码执行漏洞简介 ThinkPHP5.0.23漏洞主要涉及远程代码执行(RCE)的安全隐患。这一漏洞的存在是因为ThinkPHP框架在底层对控制器名的过滤不够严格,导致攻击者有可能通过特定的URL构造,调用到框架内部的敏感函数,进而执行任意命令。 2 thinphp5.0.23漏…

安装或者更新VMware提示:

the file is not avalid installation package for the product Microsoft Visual C 2015 x86 Minimum Runtime Try to find the installation pakcage vc_runtimeMinimum_x86.msi in a folder from which you can install Microsoft Visual C 2015 x86 Minimum Runtime 解决方…

Baidu comate智能编程助手评测

Baidu comate智能编程助手评测 作者&#xff1a;知孤云出岫 目录 一&#xff0e; 关于comate产品 二&#xff0e; 关于comate产品体验 三&#xff0e; 关于实际案例. 四&#xff0e; 关于baidu comate编程助手的实测体验感悟 五&#xff0e; …

与健康码一起落寞的,还有不被投资者待见的圣湘生物们

&#xff08;题图&#xff09; 文&#xff5c;新熔财经 作者&#xff5c;宏一 都2024年了&#xff0c;没想到还有用到健康码的一天。 前段时间&#xff0c;我咳嗽比较严重&#xff0c;去当地医院挂号做了个肺部CT&#xff0c;结果医生居然让我出示健康码&#xff0c;让我感…

力扣HOT100 - 994. 腐烂的橘子

解题思路&#xff1a; 因为要记录轮数&#xff08;分钟数&#xff09;&#xff0c;所以不能一口气遍历到底&#xff0c;所以不能用深搜&#xff08;bfs&#xff09;&#xff0c;而要用广搜&#xff08;bfs&#xff0c;层序遍历&#xff09;。 先记录下新鲜橘子数&#xff0c;…

2024年618有哪些数码家电值得入手?全网最省钱攻略指南

作为全年唯一设在夏季的大型电商狂欢节&#xff0c;618一直是很多人购置数码类、家电类的最好时间节点之一。但是问题来了&#xff0c;现在的数码家电行业“鱼龙混杂”&#xff0c;不仅越来越多新品牌涌入市场&#xff0c;而且各个大品牌为了抢占市场&#xff0c;旗下产品的品类…

新手Pytorch入门笔记-transforms.Compose()

我使用的图片是上图&#xff0c;直接下载即可 transforms.Compose 是PyTorch中的一个实用工具&#xff0c;用于创建一个包含多个数据变换操作的变换对象。这些变换操作通常用于数据预处理&#xff0c;例如图像数据的缩放、裁剪、旋转等。使用transforms.Compose 可以将多个数据…

用了十年以上的破旧电脑还能干啥?Ubuntu 24.04

用了十年之久的电脑卡不卡&#xff1f; 写博客&#xff0c;刷视频一点都不卡&#xff0c;敲敲代码也OK。 宏碁Aspire 4741G系列笔记本电脑是宏碁在2010年8月推出的产品&#xff0c;因此其生产日期可能大致在那个时间段或附近。但请注意&#xff0c;具体的生产日期可能会因生产…

Linux多进程(四) 守护进程

守护进程&#xff08;Daemon Process&#xff09;&#xff0c;也就是通常说的 Daemon 进程&#xff08;精灵进程&#xff09;&#xff0c;是 Linux 中的后台服务进程。它是一个生存期较长的进程&#xff0c;通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件…

【Android】 四大组件详解之广播接收器、内容提供器

目录 前言广播机制简介系统广播动态注册实现监听网络变化静态注册实现开机自启动 自定义广播发送标准广播发送有序广播 本地广播 内容提供器简介运行时权限访问其他程序中的数据ContentResolver的基本用法读取系统联系人 创建自己的内容提供器创建内容提供器的步骤 跨程序数据共…

尺取法知识点讲解

一、固定长度的情况&#xff1a; 最小和(sum) 输入N个数的数列&#xff0c;所有相邻的M个数的和共有N-M1个&#xff0c;求其中的最小值。 输入格式 第1行&#xff0c;2个整数N&#xff0c;M&#xff0c;范围在[3…100000]&#xff0c;N>M。 第2行&#xff0c;有N个正…

【算法刷题 | 贪心算法03】4.25(最大子数组和、买卖股票的最佳时机|| )

文章目录 4.最大子数组和4.1题目4.2解法一&#xff1a;暴力4.2.1暴力思路4.2.2代码实现 4.3解法二&#xff1a;贪心4.3.1贪心思路4.3.2代码实现 5.买卖股票的最佳时机||5.1题目5.2解法&#xff1a;贪心5.2.1贪心思路5.2.2代码实现 4.最大子数组和 4.1题目 给你一个整数数组 n…