C语言——指针进阶(2)

在这里插入图片描述
继续上次的指针,想起来还有指针的内容还没有更新完,今天来补上之前的内容,上次我们讲了函数指针,并且使用它来实现一些功能,今天我们就讲一讲函数指针数组等内容,废话不多说,我们开始今天的学习吧。

函数指针数组
在引出函数指针的时候,我们会先引出字符指针还有整型指针,那同样的道理,函数指针数组是什么,我们可能还是比较陌生,但是如果是字符指针数组,还有整型指针数组我们可能听起来不是特别陌生。
那再讲之前,我给大家先讲一下函数指针,函数指针是啥呢,函数指针竟然是指针,那就是来存放函数的地址的,那我们先写一个简单的函数,然后来存放它的地址。

int Add(int x, int y)
{return x + y;
}
int main()
{printf("%p\n", &Add);printf("%p\n", Add);return 0;
}

我们可以发现打印出来的两个地址都是同一个地址,所以表示两个方式都可以取出函数的地址

在这里插入图片描述
在这里我们可以看到我们的地址是一样的,那我们要存放他们的地址改怎么写呢

我们来看看

#include<stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{printf("%p\n", &Add);printf("%p\n", Add);int (*pf)(int, int) = &Add;return 0;
}

这样我们就可以把它函数的地址写出来,这和那个数组指针也特别相似,取出的数组地址也是这样的。
因为时间太长,我帮大家在回忆一下我们的数组指针,比如我们现在要取出的是数组的地址,我们改怎么表示,让我们来看一看吧。

int main()
{int arr[] = { 1,2,3,4,5,6 };int(*pa)[6] = &arr;return 0;
}

在这里我们可以看到的是数组指针,是指向数组地址的指针,可以看到它的格式其实和函数指针的差不多。
那我们的函数指针有什么用呢,答案是可以调用这个函数,比如就我们上面的Add函数,我们来看代码。

int Add(int x, int y)
{return x + y;
}
int main()
{printf("%p\n", &Add);printf("%p\n", Add);int (*pf)(int, int) = &Add;return 0;
}

我们可以这样写

int Add(int x, int y)
{return x + y;
}
int main()
{printf("%p\n", &Add);printf("%p\n", Add);int (*pf)(int, int) = &Add;int ret = (*pf)(3, 5);printf("%d\n", ret);return 0;
}

在这里插入图片描述
在这里其实可以不写,意思就是int ret = (*pf)(3, 5);可以写成int ret = pf(3, 5);,其实这两个的结果是一样的,或者说就算我们加很多也不影响我们的结果,来看看吧

int Add(int x, int y)
{return x + y;
}
int main()
{printf("%p\n", &Add);printf("%p\n", Add);int (*pf)(int, int) = &Add;int ret = (*pf)(3, 5);printf("%d\n", ret);ret = printf("%d\n", (*********pf)(4, 6));return 0;
}

写了这么多主要是为了后面做铺垫,那我们现在才来讲我们的函数指针数组
数组是什么,我们之前见过字符指针数组,他是来存放字符指针,我们也讲过整型指针数组,它是来存放整型指针的数组,那同样函数指针数组就是来存放函数指针的,我们来举个列子


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 (*pf1)(int, int) = &Add;int (*pf2)(int, int) = &Sub;int (*pf3)(int, int) = &Mul;int (*pf4)(int, int) = &Div;return 0;
}

我们首先看到这么多的函数,他们的功能类似我们的计算器,这里就只写了加减乘除的功能,通过我们的代码我们可以看到的是四个一样的函数指针,那我们是不能就可以用数组的形似来存放它们。因为数组是不是能存放多个类型相同的元素。
那我们就可以来写一写看

int main()
{int (*pf1)(int, int) = &Add;int (*pf2)(int, int) = &Sub;int (*pf3)(int, int) = &Mul;int (*pf4)(int, int) = &Div;int(*pfarr[4])(int, int) = { &Add,&Sub,&Mul,&Div };//pfarr就是存放函数指针的数组return 0;
}

那函数指针数组难道就是一个摆设吗,不!!!它还是有一点作用的,那它的作用是什么呢,让我们来一起看看吧。
我们是不是就可以用它来实现一个计算器,我们来设计一个吧

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 menu()
{printf("***1.Add  2.Sub ****\n");printf("***3.Mul  4.Div ****\n");printf("****0.exit      ****\n");printf("********************\n");
}
int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择>");scanf("%d" ,&input);switch (input){case 0:printf("退出计算器\n");break;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;default:printf("选择错误,请重新选择\n");break;}} while (input);return 0;}

虽然我们实现一个简单的计算器,但是其实我们的代码看起来并不是特别好看,反而有点冗余,重复的内容有点多,而且如果我们加一些其他的运算的时候,函数会变得越来越长,那我们该怎么改改让我们的代码看起来又好看又实用呢,这个时候我们就可以用函数指针数组进行改正,现在我们来对它进行整改。

我们用这个的时候之前一定要注意的问题是计算器这个函数的类型是一样的,否则其他的话就不行了。

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 menu()
{printf("***1.Add  2.Sub ****\n");printf("***3.Mul  4.Div ****\n");printf("****0.exit      ****\n");printf("********************\n");
}
int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择>");scanf("%d", &input);int (*pf[5])(int, int) = { NULL,Add,Sub,Mul,Div };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);}} while (input);return 0;}

这样也能实现我们计算器的效果,而且好处也是特别大,首先解决了我们的代码冗余的现象,其次是我们如果想加入其他的运算,也可以直接加入,只要改一下数组的大小,并把函数的地址放到里面就可以解决问题,其实很简单。我们也将这个东西叫做转移表,这就是我们函数数组指针应用,但是这只是一个例子,其他还有很多的作用,这里就不扩展了。

指向函数指针数组的指针

一看这个标题不知道大家头是不是大了,反正小编一开始看这个有点云里雾里的感觉,但是经过一段时间的学习,发现它也就这啊,那现在我们开始举列子。

int a = 10;
int b = 20;
int c = 30;
int* arr[] = { &a,&b,&c };//整型指针数组

我们先来看简单的,然后慢慢的引入,先是整型指针数组,那如果是这样的一个问题,要指向整型指针数组的指针,我们该怎么写

int* (*parr)[3] = &arr;

这就是指向整型指针数组的指针,同样的道理我们来看指向函数指针数组的指针。

int (*pf[5])(int, int) = { NULL,Add,Sub,Mul,Div };int(*(*ppf)[5])(int, int) = &pf;

我们就拿计算器的来写,就是这个样子的。
但是这个其实不用理解太深,稍微了解一下就知道了,不是特别重要。

回调函数

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

什么意思呢,就是假设我们有两个函数一个函数叫A,一个函数叫B,我们B函数的参数是A函数的地址,我们在B函数里使用这个函数A。
今天就讲一个例子,还是我们的计算器,下次给大家讲一个qsort,这也叫快速排序,让大家更好的理解我们的内容。

先把我们的计算器在拿出来给大家看一下

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 menu()
{printf("***1.Add  2.Sub ****\n");printf("***3.Mul  4.Div ****\n");printf("****0.exit      ****\n");printf("********************\n");
}
int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择>");scanf("%d" ,&input);switch (input){case 0:printf("退出计算器\n");break;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;default:break;}} while (input);return 0;}

我们可以把相同的部分也就是case语句分装成一个函数的思路去实现

我们可以看到我们的代码哪里是不太一样的地方,其实就是我们调用函数的地方,如果我们能写一个函数calc,来放这些函数的地址,问题是不是解决了,其实就是写成这个样子。

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 menu()
{printf("***1.Add  2.Sub ****\n");printf("***3.Mul  4.Div ****\n");printf("****0.exit      ****\n");printf("********************\n");
}
int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择>");scanf("%d" ,&input);switch (input){case 0:printf("退出计算器\n");break;case 1:calc(Add);break;case 2:calc(Sub);break;case 3:calc(Mul);break;case 4:calc(Div);break;default:break;}} while (input);return 0;}

那现在要做的是什么,就是完成我们的calc函数这样就可以实现计算器的功能

calc我们的返回类型就是void 函数他们的地址,那我们是不是可以写成int (*pf)(int,int)


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 menu()
{printf("***1.Add  2.Sub ****\n");printf("***3.Mul  4.Div ****\n");printf("****0.exit      ****\n");printf("********************\n");
}void calc(int (*pf)(int, int))
{int x = 0;int y = 0;int ret = 0;printf("请输入两个操作数 :");scanf("%d %d", &x, &y);ret = pf(x, y);printf("ret = %d\n", ret);
}
int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择>");scanf("%d", &input);switch (input){case 0:printf("退出计算器\n");break;case 1:calc(Add);break;case 2:calc(Sub);break;case 3:calc(Mul);break;case 4:calc(Div);break;default:break;}} while (input);return 0;}

这样就把之前的功能也实现了。我们后面再讲一个qsort,今天懒得讲了,摆烂ing

我们下次再见,拜拜

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

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

相关文章

页面分布引导新手指引(driver.js)

页面分布引导&#xff08;driver.js&#xff09; 最近由于有一个需求——做新手指引&#xff0c;在新用户进入页面的时候提供指引和帮助,快速让用户熟悉页面的功能,但是为了不要过多影响现有的页面逻辑和样式,找到一款非常好用的工具driver.js:Driver.js是一个功能强大且高度可…

成都瀚网科技有限公司:抖音商家怎么免费入驻?

随着抖音成为全球最受欢迎的短视频平台之一&#xff0c;越来越多的商家开始关注抖音上的商机。抖音商家的进驻可以帮助商家扩大品牌影响力和销售渠道。那么&#xff0c;如何免费进入抖音成为商家呢&#xff1f;下面就为大家介绍一下具体步骤。 1、抖音商家如何免费注册&#xf…

Mojo安装使用初体验

一个声称比python块68000倍的语言 蹭个热度&#xff0c;安装试试 系统配置要求&#xff1a; 不支持Windows系统 配置要求: 系统&#xff1a;Ubuntu 20.04/22.04 LTSCPU&#xff1a;x86-64 CPU (with SSE4.2 or newer)内存&#xff1a;8 GiB memoryPython 3.8 - 3.10g or cla…

C++斩题录|递归专题 | leetcode50. Pow(x, n)

个人主页&#xff1a;平行线也会相交 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 平行线也会相交 原创 收录于专栏【手撕算法系列专栏】【LeetCode】 &#x1f354;本专栏旨在提高自己算法能力的同时&#xff0c;记录一下自己的学习过程&#xff0c;希望…

Discourse 可以支持的存储类型

根据官方的这个主题&#xff1a;Configure an S3 compatible object storage provider for uploads - sysadmin - Discourse Meta Discourse 可以支持很多不同的对象存储。 感觉上是只要和 S3 兼容的基本上都能用。 建议 从对象存储的角度考虑&#xff0c;还是建议使用 S3。…

C# Linq源码分析之Take(五)

概要 本文在C# Linq源码分析之Take&#xff08;四&#xff09;的基础上继续从源码角度分析Take的优化方法&#xff0c;主要分析Where.Select.Take的使用案例。 Where.Select.Take的案例分析 该场景模拟我们显示中将EF中与数据库关联的对象进行过滤&#xff0c;然后转换成Web…

代码随想录第30天 | ● 860.柠檬水找零 ● 406.根据身高重建队列 ● 452. 用最少数量的箭引爆气球

860.柠檬水找零 /*** param {number[]} bills* return {boolean}*/ var lemonadeChange function(bills) {let d50let d100let d200for(let i0;i<bills.length;i){if(bills[i]5){d51}else if(bills[i]10){if(d5>1){d5--d10}elsereturn false}else{if(d5>1&&…

表情识别-情感分析-人脸识别(代码+教程)

表情识别 面部情绪识别&#xff08;FER&#xff09;是指根据面部表情识别和分类人类情绪的过程。通过分析面部特征和模式&#xff0c;机器可以对一个人的情绪状态作出有根据的推断。这个面部识别的子领域高度跨学科&#xff0c;涉及计算机视觉、机器学习和心理学等领域的知识。…

Java输入-a,-b,geek,-c,888,-d,[hello,world]字符之后,如何将[hello,world]这个不分开

Java输入-a,-b,geek,-c,888,-d,[hello,world]字符之后&#xff0c;如何将[hello,world]这个不分开&#xff1f; 你可以使用命令行参数解析库来处理Java输入中的各个参数。在这种情况下&#xff0c;你可以使用Apache Commons CLI库来解析命令行参数。以下是一个示例代码片段&am…

Day60|单调栈part03:84.柱状图中最大的矩形

柱状图中最大的矩形 leetcode链接&#xff1a;力扣题目链接 视频链接&#xff1a;单调栈&#xff0c;又一次经典来袭&#xff01; LeetCode&#xff1a;84.柱状图中最大的矩形 给定 n 个非负整数&#xff0c;用来表示柱状图中各个柱子的高度。每个柱子彼此相邻&#xff0c;…

3分钟:腾讯云免费SSL证书申请教程_免费HTTPS证书50张

2023腾讯云免费SSL证书申请流程&#xff0c;一个腾讯云账号可以申请50张免费SSL证书&#xff0c;免费SSL证书为DV证书&#xff0c;仅支持单一域名&#xff0c;申请腾讯云免费SSL证书3分钟即可申请成功&#xff0c;免费SSL证书品牌为TrustAsia亚洲诚信&#xff0c;腾讯云百科分享…

2023国赛数学建模B题思路代码 - 多波束测线问题

# 1 赛题 B 题 多波束测线问题 单波束测深是利用声波在水中的传播特性来测量水体深度的技术。声波在均匀介质中作匀 速直线传播&#xff0c; 在不同界面上产生反射&#xff0c; 利用这一原理&#xff0c;从测量船换能器垂直向海底发射声波信 号&#xff0c;并记录从声波发射到…

透视俄乌网络战之二:Conti勒索软件集团(上)

透视俄乌网络战之一&#xff1a;数据擦除软件 Conti勒索软件集团&#xff08;上&#xff09; 1. Conti简介2. 组织架构3. 核心成员4. 招募途径5. 工作薪酬6. 未来计划参考 1. Conti简介 Conti于2019年首次被发现&#xff0c;现已成为网络世界中最危险的勒索软件之一&#xff0…

HashMap、LinkedHashMap、ConcurrentHashMap、ArrayList、LinkedList的底层实现。

HashMap、LinkedHashMap、ConcurrentHashMap、ArrayList、LinkedList的底层实现。 HashMap相关问题 1、你用过HashMap吗&#xff1f;什么是HashMap&#xff1f;你为什么用到它&#xff1f;用过&#xff0c;HashMap是基于哈希表的Map接口的非同步实现&#xff0c; 它允许null键…

css重置

css 重置 CSS 重置的主要目标是确保浏览器之间的一致性&#xff0c;并撤消所有默认样式&#xff0c;创建一个空白板。 如今&#xff0c;主流浏览器都实现了css规范&#xff0c;在布局或间距方面没有太大差异。但是通过自定义 CSS 重置&#xff0c;也可以改善用户体验和提高开…

面试问题总结(1)

(꒪ꇴ꒪ )&#xff0c;Hello我是祐言QAQ我的博客主页&#xff1a;C/C语言&#xff0c;数据结构&#xff0c;Linux基础&#xff0c;ARM开发板&#xff0c;网络编程等领域UP&#x1f30d;快上&#x1f698;&#xff0c;一起学习&#xff0c;让我们成为一个强大的攻城狮&#xff0…

UMA 2 - Unity Multipurpose Avatar☀️八.UMA内置实用Recipes插件

文章目录 🟥 UMA内置Recipes位置🟧 CapsuleCollider🟨 Expressions : 表情管理(重点)🟩 Locomotion : 移动测试的插件🟦 Physics : Collider升级版🟥 UMA内置Recipes位置 如下图所示,UMA共内置5种实用Recipes,文件夹内的Text Recipes类型的文件即是实用Recipes. …

mac帧 arp

1.分片 2.MSS max segment size 3.跨网络的本质 就是经历很多的子网或者局域网 4.将数据从A主机跨网络送到B主机的能力 IP和mac IP解决的是路径选择的问题 5.数据链路层 用于两个设备&#xff08;同一种数据链路节点&#xff09;之间进行传递 6.以太网ether 7.局域网通…

初学python(一)

一、python的背景和前景 二、 python的一些小事项 1、在Java、C中&#xff0c;2 / 3 0&#xff0c;也就是整数 / 整数 整数&#xff0c;会把小数部分舍掉。而在python中2 / 3 0.66666.... 不会舍掉小数部分。 在编程语言中&#xff0c;浮点数遵循IEEE754标准&#xff0c;不…

Spark【RDD编程(二)RDD编程基础】

前言 接上午的那一篇&#xff0c;下午我们学习剩下的RDD编程&#xff0c;RDD操作中的剩下的转换操作和行动操作&#xff0c;最好把剩下的RDD编程都学完。 Spark【RDD编程&#xff08;一&#xff09;RDD编程基础】 RDD 转换操作 6、distinct 对 RDD 集合内部的元素进行去重…