转自:https://segmentfault.com/a/1190000008293902?utm_source=tag-newest

在面试的时候被问到什么是回调函数,我是属于会用但不懂概念的那类,即知其然不知其所以然。特查询见此篇文章解释很清晰,故转载保留。

什么是回调函数

我们先来看看百度百科是如何定义回调函数的:

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

这段话比较长,也比较绕口。下面我通过一幅图来说明什么是回调:

假设我们要使用一个排序函数来对数组进行排序,那么在主程序(Main program)中,我们先通过库,选择一个库排序函数(Library function)。但排序算法有很多,有冒泡排序,选择排序,快速排序,归并排序。同时,我们也可能需要对特殊的对象进行排序,比如特定的结构体等。库函数会根据我们的需要选择一种排序算法,然后调用实现该算法的函数来完成排序工作。这个被调用的排序函数就是回调函数(Callback function)。

结合这幅图和上面对回调函数的解释,我们可以发现,要实现回调函数,最关键的一点就是要将函数的指针传递给一个函数(上图中是库函数),然后这个函数就可以通过这个指针来调用回调函数了。注意,回调函数并不是C语言特有的,几乎任何语言都有回调函数。在C语言中,我们通过使用函数指针来实现回调函数。那函数指针是什么?不着急,下面我们就先来看看什么是函数指针。

什么是函数指针

函数指针也是一种指针,只是它指向的不是整型,字符型而是函数。在C中,每个函数在编译后都是存储在内存中,并且每个函数都有一个入口地址,根据这个地址,我们便可以访问并使用这个函数。函数指针就是通过指向这个函数的入口,从而调用这个函数。

函数指针的使用

函数指针的定义

函数指针虽然也是指针,但它的定义方式却和其他指针看上去很不一样,我们来看看它是如何定义的:

  1. /* 方法1 */
  2. void (*p_func)(int, int, float) = NULL;
  3. /* 方法2 */
  4. typedef void (*tp_func)(int, int, float);
  5. tp_func p_func = NULL;

这两种方式都是定义了一个指向返回值为 void 类型,参数为 (int, int, float) 的函数指针。第二种方法是为了让函数指针更容易理解,尤其是在复杂的环境下;而对于一般的函数指针,直接用第一种方法就行了。
如果之前没见过函数指针,可能会觉得函数指针的定义比较怪,为什么不是 void ()(int, int, float) *p_func 而是 void (*p_func)(int, int, float) 这种形式?这个问题我也不知道,也没必要纠结,花点时间理解下它与普通指针的区别,实在不行就先记住它的形式。

函数指针的赋值

在定义完函数指针后,我们就需要给它赋值了我们有两种方式对函数指针进行赋值:

  1. void (*p_func)(int, int, float) = NULL;
  2. p_func = &func1;
  3. p_func = func2;

上面两种方法都是合法的,对于第二种方法,编译器会隐式地将 func_2 由 void ()(int, int, float) 类型转换成 void (*)(int, int, float) 类型,因此,这两种方法都行。

使用函数指针调用函数

因为函数指针也是指针,因此可以使用常规的带 * 的方法来调用函数。和函数指针的赋值一样,我们也可以使用两种方法:

  1. /* 方法1 */
  2. int val1 = p_func(1,2,3.0);
  3. /* 方法2 */
  4. int val2 = (*p_func)(1,2,3.0);

方法1和我们平时直接调用函数是一样的,方法2则是用了 * 对函数指针取值,从而实现对函数的调用。

将函数指针作为参数传给函数

函数指针和普通指针一样,我们可以将它作为函数的参数传递给函数,下面我们看看如何实现函数指针的传参:

  1. /* func3 将函数指针 p_func 作为其形参 */
  2. void func3(int a, int b, float c, void (*p_func)(int, int, float))
  3. {
  4. (*p_func)(a, b, c);
  5. }
  6. /* func4 调用函数func3 */
  7. void func4()
  8. {
  9. func3(1, 2, 3.0, func_1);
  10. /* 或者 func3(1, 2, 3.0, &func_1); */
  11. }

函数指针作为函数返回类型

有了上面的基础,要写出返回类型为函数指针的函数应该不难了,下面这个例子就是返回类型为函数指针的函数:

  1. void (* func5(int, int, float ))(int, int)
  2. {
  3. ...
  4. }

在这里, func5 以 (int, int, float) 为参数,其返回类型为 void (*)(int, int) 。

函数指针数组

在开始讲解回调函数前,最后介绍一下函数指针数组。既然函数指针也是指针,那我们就可以用数组来存放函数指针。下面我们看一个函数指针数组的例子:

  1. /* 方法1 */
  2. void (*func_array_1[5])(int, int, float);
  3. /* 方法2 */
  4. typedef void (*p_func_array)(int, int, float);
  5. p_func_array func_array_2[5];

上面两种方法都可以用来定义函数指针数组,它们定义了一个元素个数为5,类型是 void (*)(int, int, float) 的函数指针数组。

回调函数

我们前面谈的都是函数指针,现在我们回到正题,来看看回调函数到底是怎样实现的。下面是一个四则运算的简单回调函数例子:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3.  
  4. /****************************************
  5. * 函数指针结构体
  6. ***************************************/
  7. typedef struct _OP {
  8. float (*p_add)(float, float);
  9. float (*p_sub)(float, float);
  10. float (*p_mul)(float, float);
  11. float (*p_div)(float, float);
  12. } OP;
  13.  
  14. /****************************************
  15. * 加减乘除函数
  16. ***************************************/
  17. float ADD(float a, float b)
  18. {
  19. return a + b;
  20. }
  21.  
  22. float SUB(float a, float b)
  23. {
  24. return a - b;
  25. }
  26.  
  27. float MUL(float a, float b)
  28. {
  29. return a * b;
  30. }
  31.  
  32. float DIV(float a, float b)
  33. {
  34. return a / b;
  35. }
  36.  
  37. /****************************************
  38. * 初始化函数指针
  39. ***************************************/
  40. void init_op(OP *op)
  41. {
  42. op->p_add = ADD;
  43. op->p_sub = SUB;
  44. op->p_mul = &MUL;
  45. op->p_div = &DIV;
  46. }
  47.  
  48. /****************************************
  49. * 库函数
  50. ***************************************/
  51. float add_sub_mul_div(float a, float b, float (*op_func)(float, float))
  52. {
  53. return (*op_func)(a, b);
  54. }
  55.  
  56. int main(int argc, char *argv[])
  57. {
  58. OP *op = (OP *)malloc(sizeof(OP));
  59. init_op(op);
  60.  
  61. /* 直接使用函数指针调用函数 */
  62. printf("ADD = %f, SUB = %f, MUL = %f, DIV = %f\n", (op->p_add)(1.3, 2.2), (*op->p_sub)(1.3, 2.2),
  63. (op->p_mul)(1.3, 2.2), (*op->p_div)(1.3, 2.2));
  64.  
  65. /* 调用回调函数 */
  66. printf("ADD = %f, SUB = %f, MUL = %f, DIV = %f\n",
  67. add_sub_mul_div(1.3, 2.2, ADD),
  68. add_sub_mul_div(1.3, 2.2, SUB),
  69. add_sub_mul_div(1.3, 2.2, MUL),
  70. add_sub_mul_div(1.3, 2.2, DIV));
  71.  
  72. return ;
  73. }

这个例子有点长,我一步步地来讲解如何使用回调函数。

第一步

要完成加减乘除,我们需要定义四个函数分别实现加减乘除的运算功能,这几个函数就是:

  1. /****************************************
  2. * 加减乘除函数
  3. ***************************************/
  4. float ADD(float a, float b)
  5. {
  6. return a + b;
  7. }
  8. float SUB(float a, float b)
  9. {
  10. return a - b;
  11. }
  12. float MUL(float a, float b)
  13. {
  14. return a * b;
  15. }
  16. float DIV(float a, float b)
  17. {
  18. return a / b;
  19. }

第二步

我们需要定义四个函数指针分别指向这四个函数:

  1. /****************************************
  2. * 函数指针结构体
  3. ***************************************/
  4. typedef struct _OP {
  5. float (*p_add)(float, float);
  6. float (*p_sub)(float, float);
  7. float (*p_mul)(float, float);
  8. float (*p_div)(float, float);
  9. } OP;
  10. /****************************************
  11. * 初始化函数指针
  12. ***************************************/
  13. void init_op(OP *op)
  14. {
  15. op->p_add = ADD;
  16. op->p_sub = SUB;
  17. op->p_mul = &MUL;
  18. op->p_div = &DIV;
  19. }

第三步

我们需要创建一个“库函数”,这个函数以函数指针为参数,通过它来调用不同的函数:

  1. /****************************************
  2. * 库函数
  3. ***************************************/
  4. float add_sub_mul_div(float a, float b, float (*op_func)(float, float))
  5. {
  6. return (*op_func)(a, b);
  7. }

第四步

当这几部都完成后,我们就可以开始调用回调函数了:

  1. /* 调用回调函数 */
  2. printf("ADD = %f, SUB = %f, MUL = %f, DIV = %f\n",
  3. add_sub_mul_div(1.3, 2.2, op->p_add),
  4. add_sub_mul_div(1.3, 2.2, op->p_sub),
  5. add_sub_mul_div(1.3, 2.2, MUL),
  6. add_sub_mul_div(1.3, 2.2, DIV));

简单的四部便可以实现回调函数。在这四步中,我们甚至可以省略第二步,直接将函数名传入“库函数”,比如上面的乘法和除法运算。回调函数的核心就是函数指针,只要搞懂了函数指针再学回调函数,那真是手到擒来了。

C回调函数的更多相关文章

  1. 小兔JS教程(三)-- 彻底攻略JS回调函数

    这一讲来谈谈回调函数. 其实一句话就能概括这个东西: 回调函数就是把一个函数当做参数,传入另一个函数中.传进去的目的仅仅是为了在某个时刻去执行它. 如果不执行,那么你传一个函数进去干嘛呢? 就比如说对 ...

  2. 嵌入式&iOS:回调函数(C)与block(OC)传 参/函数 对比

    C的回调函数: callBack.h 1).声明一个doSomeThingCount函数,参数为一个(无返回值,1个int参数的)函数. void DSTCount(void(*CallBack)(i ...

  3. 嵌入式&iOS:回调函数(C)与block(OC)回调对比

    学了OC的block,再写C的回调函数有点别扭,对比下区别,回忆记录下. C的回调函数: callBack.h 1).定义一个回调函数的参数数量.类型. typedef void (*CallBack ...

  4. 理解 JavaScript 回调函数并使用

    JavaScript中,函数是一等(first-class)对象:也就是说,函数是 Object 类型并且可以像其他一等对象(String,Array,Number等)一样使用.它们可以"保 ...

  5. 关于js的回调函数的一点看法

    算了一下又有好几个月没写博客了,最近在忙公司android的项目,所以也就很少抽时间来写些东西了.刚闲下来,我就翻了翻之前看的东西.做了android之后更加感觉到手机端开发的重要性,现在做nativ ...

  6. JS学习:第二周——NO.1回调函数

    [回调函数] 定义:把一个函数的定义阶段,作为参数,传给另一个函数: 回调函数调用次数,取决于条件: 回调函数可以传参: 回调函数可以给变this指向,默认是window: 回调函数没有返回值,for ...

  7. 【java回调】java两个类之间的回调函数传递

    背景交代:熟悉用js开发的cordovaAPP:对java一窍不通的我,老师让做一个监测用户拍照事件的功能,无奈没有找到现成的库,无奈自己动手开发java插件~~0基础java GreenHand,祝 ...

  8. Java|今天起,别再扯订阅和回调函数

    编程史上有两个令人匪夷所思的说辞,一个是订阅,一个是回调函数. 我想应该还有很多同学为“事件的订阅”和“回调函数”所困扰,因为事情本来就不应该按这个套路来解释. 多直白,所谓的“回调函数”你完全可以线 ...

  9. C++ 回调函数的定义与用法

    一回调函数 我们经常在C++设计时通过使用回调函数可以使有些应用(如定时器事件回调处理.用回调函数记录某操作进度等)变得非常方便和符合逻辑,那么它的内在机制如何呢,怎么定义呢?它和其它函数(比如钩子函 ...

  10. 通过修改i8042prt端口驱动中类驱动Kbdclass的回调函数地址,达到过滤键盘操作的例子

    同样也是寒江独钓的例子,但只给了思路,现贴出实现代码 原理是通过改变端口驱动中本该调用类驱动回调函数的地方下手 //替换分发函数 来实现过滤 #include <wdm.h> #inclu ...

随机推荐

  1. CircRNA 环化RNA

    2016国自然新秀CircRNA的研究策略和分析  

  2. windows server2012安装mysql时一直停留在start server的解决方法

    安装的时候,starting server 卡住 原因分析 这个问题小编安装mysql时也碰到过,出现这个问题是my.ini文件没有复制成功了,我们只要在mysql安装目录把把目录中的备份的my-sm ...

  3. Locality preserving hashing for fast image search: theory and applications

    Is there any Java library that provides an implementation (or several) of a Locality Preserving Hash ...

  4. Acrobat_8_Pro_SC 激活老是提示你输入的授权码无效

    假如安装了Adobe Acrobat Professional 8 的时候无法激活, 或在恢复安装时 Adobe Acrobat Professional 8 需要重新激活, 激活的时候,总是提示你输 ...

  5. VCS简介

    VCS -Version Control System 版本控制是一种记录一个或若干文件内容变化,以便将来查阅特定版本修订情况的系统. 特征 1.记录文件的所有历史变化 2.随时可恢复到任何一个历史状 ...

  6. XE5安卓手机要求

    1 ARMv7 的 CPU v6 的肯定不支持.2 黑屏是因为你的手机 CPU 不支持 NEON 特性.或者是 T2 CPU.3 系统版本  2.3.3 到 2.3.9 或者 4.0 以上.4. SD ...

  7. ResorceGovernor--基础和Demo

    资源调控器分为三部分:1:资源池,将资源CPU/MEMORY划分到不同的载体上2:负载组,承载负载并将负载映射到不同的资源池3: 分类函数,将不同回话映射到不同的负载组08提供两种预定义的系统资源池1 ...

  8. c# winform实现同时只允许账号在一台电脑登录的功能

    公司有个小需求,要求账号不能同时登录在多台电脑上,就像那个微信或QQ一样,如果一台电脑登录了,原来登录的就自动退出了(网上搜索点单点登录,发现有些出入,人家是实现一次登录在多个系统间认证的) 找了些资 ...

  9. AJAX方式调用百度天气

    后台代码: [HttpPost] public string AjaxWeather() { string CityName = string.IsNullOrEmpty(Request.Form[& ...

  10. 【《Effective C#》提炼总结】提高Unity中C#代码质量的22条准则

    引言 原则1尽可能地使用属性而不是可直接访问的数据成员 原则2偏向于使用运行时常量而不是编译时常量 原则3 推荐使用is 或as操作符而不是强制类型转换 原则4 推荐使用条件属性而不是if条件编译 原 ...