【C语言进阶深度学习记录】三十二 函数指针与使用函数指针实现回调函数

回调函数是非常重要的概念

文章目录

    • 1 函数的类型
    • 2 函数指针
      • 2.1 函数指针的使用
      • 2.2 使用函数指针实现回调函数
    • 3 总结

1 函数的类型

跟以前学数组的时候是一样的,C语言中的数组是有自己的类型的。函数也是有自己的类型的。

  • 函数的类型由返回值、参数的类型、参数的个数共同决定

比如函数int add(int i, int j); 的函数类型是int (int, int)

  • 在C语言中可以使用typedef为函数重命名,如下
   typedef int f(int, int); // 定义f为函数类型int(int, int)typedef void p(int);   //   定义p为函数类型void(int)

上面定义函数类型一会再后面写具体的代码就会明白。

2 函数指针

与数组指针很相似

  • 函数指针用于指向一个函数
  • 函数名执行函数体的入口地址(这里与数组指针一样,数组名代表数组的入口地址)
  • 可以通过函数的类型定义函数指针:FuncType* pointer(例如上面定义的f,那么f* p; p指向函数类型为f的函数。)
  • 也可以直接定义(这种我们见到的比较多):type (*pointer)(parameter list)

上面:

  1. FuncType 是一种函数类型(例如int (int, int))
  2. type是函数的返回类型
  3. pointer是函数指针
  4. parameter list 是函数的参数列表

2.1 函数指针的使用

下面代码是演示定义函数类型与如何使用函数指针的代码:

  • 36-1.c:
#include <stdio.h>typedef int FUNC(int);  //定义函数类型int t(int i){return i*i;
}void f(){printf("Call f()....\n");
}int main(){FUNC* pt = t;  //定义函数指针ptvoid(*pf)() = &f;  //定义函数指针pfprintf("pf = %p\n", pf);printf("f = %p\n", f);printf("&f = %p\n", &f);pf();   //一般这么用(*pf)();  //一般不这么用f();int k = pt(3);printf("k = %d\n", k);return 0;
}
  • 运行结果为:

在这里插入图片描述

上面代码比较简单,我们注意两点:

  • typedef int FUNC(int); //定义函数类型
  • FUNC* pt = t; //定义函数指针pt
  • void(*pf)() = &f; //定义函数指针pf
  • 函数名f代表函数的入口地址,&f代表函数的整个地址(类似于数组),所以在数值上f与&f是相等的。但是意义不一样。
  • 函数指针的使用:pf(); 或者 (*pf)(); 一般像前者那么使用

2.2 使用函数指针实现回调函数

如何使用C语言直接跳转到某个固定的地址开始执行?(不是goto)

使用回调函数,回调函数是一种很重要的思想。那么什么是回调机制呢?

  • 回调机制原理:
  1. 调用者不知道具体事件发生时需要调用哪一个具体的函数 (是不是与C++中的多态很像,是的C++中的多态原理就是这样)
  2. 具体的事件发生的时候 调用者通过函数指针,调用具体的函数。(是不是很像C++中的虚函数指针)

回调机制中的调用者和被调用者没有依赖关系

  • 上面的理论,看起来并不是很容易让人理解,下面直接上手写代码就知道什么是回调函数了。

  • 代码 36-2.c

#include <stdio.h>typedef int (Fruit)(int); //定义Fruit为int(int)类型的函数 参数:吃多少克,返回值:获得多少能量void Eat(Fruit* fruit, int n){ //函数指针fruit:指向吃什么水果的函数  参数n:吃多少克int ret = 0;printf("Eat...\n");ret = fruit(n);    //获得多少能量printf("Increase : %d\n", ret);
}int Apple(int n){    //吃n克苹果获得ret克能量int ret = 0;int i = 0;for(i=0; i<n; i++){printf("Eat apple get energy : %d\n", 1);ret++;}return ret;
}int Banana(int n){   //吃n克香蕉获得ret克能量int ret = 0;int i =0;for(i=0; i<n; i++){printf("Eat banana get energy : %d\n",3);ret+=3;}return ret;
}int Pear(int n){     //吃n克梨子获得ret克能量int ret = 0;int i = 0;for(i=0; i<n; i++){printf("Eat pear get energy : %d\n",5);ret+=5;}return ret;
}
int main(){Eat(Apple, 5);  //get 5   printf("\n");Eat(Banana, 2);  // get 6printf("\n");Eat(Pear, 3);   //get 15printf("\n");return 0;
}
  • 上述代码的意思是:吃水果获得能量。但是吃什么水果,只有在程序运行起来之后才知道。
  • 所以吃这个动作Eat函数的参数无法指定吃哪种水果
  • 只能使用函数指针作为Eat函数的参数,当程序运行起来时根据传进来的参数确定吃什么水果,以及获得多少能量

上述程序编译运行结果为:

在这里插入图片描述

分析:

  • 上面代码主要的核心就在于函数指针的使用。一定要学会函数指针的定义使用。
  • 学会使用函数指针来实现回调函数

3 总结

  • 学会定义函数类型
  • 学会使用函数类型定义函数指针
  • 学会使用函数指针实现回调机制

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

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

相关文章

【C语言进阶深度学习记录】三十三 C语言中动态内存分配

如何在程序运行的时候动态给程序分配内存&#xff1f; 文章目录1 动态内存分配的意义1.1 C语言中如何动态申请内存空间1.2 malloc和free的用法1.3 calloc与realloc1.31 calloc和realloc的代码案例分析2 总结1 动态内存分配的意义 在C语言中&#xff0c;一切操作都是基于内存的…

java并发实战

推荐一个Java并发编程实战的学习专栏。此专栏为极客时间收费专栏。 学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群 &#xff1a; 962535112 对于一个 Java 程序员而言&#xff0c; 能否熟练掌握并发编程是判断他优秀与否的…

a critical review of preetham skylight model 笔记

也是为了试用下Xmind。 上图是我用Xmind作的某篇文章的笔记。 感想&#xff1a; 1. 之以一直觉得这种东西没多大用处&#xff0c;回想起来大概是因为那时没有太多应用场景。 2. 如果留心&#xff0c;可以把许多事情做得更漂亮、更容易&#xff0c;这也是工具的用途。 贴一下软件…

【C语言进阶深度学习记录】三十四 C语言实现内存泄漏检测模块

上一篇文章学习了malloc系列的三个函数的使用。众所周知malloc的使用很容易导致内存泄漏。本文的目的就是使用C语言来实现内存泄漏检测模块&#xff0c;来帮忙自动检测我们写的程序中是否出现内存泄露。 文章目录1 内存泄露检测模块的实现原理1.1 各个函数模块的设计1.2 模块整…

重学前端----前端知识系统学习推荐专栏

推荐一个前端知识学习专栏。此专栏为极客时间收费专栏。 学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 文章目录1 概述2 专栏内容2.1 模块一&#xff0c;JavaScript2.2 模块二&#xff0c;HTML 和 …

【C语言进阶深度学习记录】三十五 程序中的堆、栈以及静态存储区(数据区)

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 在我之前学习底层的知识的时候&#xff0c;也写过相关的内容。可以对比的学习&#xff1a;【软件开发底层知识修炼】二十 深入理解可执行程序的结构&a…

【C语言进阶深度学习记录】三十六 程序与进程的区别(程序的内存布局)

上一篇文章学了堆&#xff0c;栈以及静态存储区。它们实际上都是针对进程来说的。那么程序与进程有什么区别呢&#xff1f; 本文不细讲程序与进程。 1 程序与进程 1.1 什么是程序 写完的.c文件是源文件。也叫源代码。 将源代码编译后&#xff0c;会生成可执行文件程序&#…

使用Cucumber+Rspec玩转BDD(2)——邮件激活

使用CucumberRspec玩转BDD(2)——邮件激活 2009年3月2日 星期一 ### 温故知新 ###前面我们已经完成了新用户注册功能的开发&#xff0c;为了方便我们后面的开发工作且不扰乱之前的工作成果&#xff0c;我们先将这份源代码归档并做个标记。为了获得更好的阅读体验&#xff0c;读…

【C语言进阶深度学习记录】三十七 C/C++中造成程序内存错误的原因(野指针)

什么是野指针&#xff1f; 指针变量存的地址是一块非法内存地址。进而形成野指针。但是需要注意一点&#xff0c;野指针不是NULL指针。 文章目录1 野指针的概念1.1 野指针代码案例初探2 如何避免野指针2.1 野指针代码案例分析进阶3 总结1 野指针的概念 野指针变量中的值是非法…

算法补充 2011-9-12

设计一个算法将顺序表L中所有小于0的整数放前半部分&#xff0c;大于等于0的整数放在后半部分二叉树的删除设计一个算法将顺序表L中所有小于0的整数放前半部分&#xff0c;大于等于0的整数放在后半部分 思路:从左侧找出>0的元素&#xff0c;从右侧找出<0的元素,然后进行交…

【C语言进阶深度学习记录】三十八 C/C++语言中的函数声明与函数定义

文章目录1 函数的声明和定义1.1 代码分析2 总结1 函数的声明和定义 声明的意义在于告诉编译器程序单元的存在。只是告诉编译器它存在但是不在声明这里定义&#xff0c;有可能在当前文件中的其他地方或者其他文件中定义。如果在它还没有被定义之前就使用它&#xff0c;会导致编…

ASP.NET MVC3 系列教程 - 部署你的WEB应用到IIS 6.0

I:ASP.NET MVC3 部署的前期工作 1.确认部署的服务器操作系统环境 首先我们确认服务器的操作系统版本可以从系统命令行工具里输入: systeminfo 获取相关操作系统信息例如然后再确认IIS版本信息 -> 打开IIS管理工具即可接着确认.NET Framework的版本可以在系统命令行工具执行:…

【C语言进阶深度学习记录】三十九 C语言中的可变参数(参数可变的函数)

用过printf()函数的热都知道&#xff0c;printf的参数可以有多个&#xff0c;它是可变的&#xff0c;根据我们输出参数的类型以及个数的不同来确定参数。今天来学习C语言中参数可变的函数是如何实现的。 文章目录1 可变参数2 总结1 可变参数 首先我们要明白一点&#xff0c;在…

【离散数学中的数据结构与算法】一 最大公约数与最小公倍数之间的关系

文章目录1 算数基本定理2 最大公约数3 最小公倍数4 性质5 推论1 算数基本定理 设正整数 n>1&#xff0c; 则 n 可唯一地表示为&#xff1a; 其中 p1<p2<,…, <ps 是 s 个相异的素数&#xff0c; 指数ki都是正整数。 此定理又称作唯一析因定理&#xff08;unique f…

【离散数学中的数据结构与算法】二 欧几里得算法与裴蜀等式

欧几里得算法是计算两个数最大公因子算法。又称辗转相除法。本文将学习为什么辗转相除法可以求得两个数的最大公因子。同时也可以根据最大公因子计算两个数的最小公倍数。 文章目录1 欧几里得算法的理论基础1.1 欧几里得算法&#xff08;辗转相除法&#xff09;2 裴蜀等式&…

【离散数学中的数据结构与算法】四 加法法则与乘法法则

文章目录1 加法法则2 乘法法则3 例子3.1 例一3.2 例二3.3 例三4 总结1 加法法则 加法法则&#xff1a; 设事件 A 有 m 种产生方式&#xff0c; 事件 B 有n 种产生方式&#xff0c;则当 A 与 B 产生的方式不重叠时&#xff0c;“事件 A 或 B 之一” 有 mn 种产生方式。 加法法…

前端学习(310):清除浮动的方法

我们经常把高度塌陷问题也叫做常见的几种清除浮动的方法 高度塌陷问题—父元素高度自适应&#xff0c;子元素float后&#xff0c;造成父元素高度为0&#xff0c;就叫做高度塌陷问题 给父元素一个高度 缺点&#xff1a;无法高度自适应 父元素{overflow:hidden;} 缺点&#xf…

【离散数学中的数据结构与算法】五 排列与组合一

在leetcode刷题过程中&#xff0c;遇到过很多关于排列组合的问题。弄清楚排列组合的相关原理&#xff0c;是非常有用处的。 文章目录1 问题2 排列-有序选取2.1 重复选取-可重排列2.2 不重复选取-排列2.21 全排列3 例题4 总结1 问题 设集合S包含n个元素&#xff0c;从S中选取r个…