回调函数例子2(C语言)

[cpp]  view plain copy
  1.    
  2. 如果参数是一个函数指针,调用者可以传递一个函数的地址给实现者,让实现者去调用它,这称为回调函数(Callback Function)。例如qsort(3)和bsearch(3)。  
  3. 表 24.7. 回调函数示例:void func(void (*f)(void *), void *p);  
  4. 调用者实现者  
  5.   
  6.  1. 提供一个回调函数,再提供一个准备传给回调函数的参数。  
  7.   
  8.  2. 把回调函数传给参数f,把准备传给回调函数的参数按void *类型传给参数p  
  9.   
  10.    
  11.   
  12.  1. 在适当的时候根据调用者传来的函数指针f调用回调函数,将调用者传来的参数p转交给回调函数,即调用f(p);  
  13.   
  14.    
  15.   
  16. 以下是一个简单的例子。实现了一个repeat_three_times函数,可以把调用者传来的任何回调函数连续执行三次。  
  17. 例 24.7. 回调函数  
  18. /* para_callback.h */  
  19. #ifndef PARA_CALLBACK_H  
  20. #define PARA_CALLBACK_H  
  21.   
  22. typedef void (*callback_t)(void *);  
  23. extern void repeat_three_times(callback_t, void *);  
  24.   
  25. #endif  
  26. /* para_callback.c */  
  27. #include "para_callback.h"  
  28.   
  29. void repeat_three_times(callback_t f, void *para)  
  30. {  
  31.      f(para);  
  32.      f(para);  
  33.      f(para);  
  34. }  
  35. /* main.c */  
  36. #include <stdio.h>  
  37. #include "para_callback.h"  
  38.   
  39. void say_hello(void *str)  
  40. {  
  41.      printf("Hello %s\n", (const char *)str);  
  42. }  
  43.   
  44. void count_numbers(void *num)  
  45. {  
  46.      int i;  
  47.      for(i=1; i<=(int)num; i++)  
  48.           printf("%d ", i);  
  49.      putchar('\n');  
  50. }  
  51.   
  52. int main(void)  
  53. {  
  54.      repeat_three_times(say_hello, "Guys");  
  55.      repeat_three_times(count_numbers, (void *)4);  
  56.      return 0;  
  57. }  
  58.   
  59. 回顾一下前面几节的例子,参数类型都是由实现者规定的。而本例中回调函数的参数按什么类型解释由调用者规定,对于实现者来说就是一个void *指针,实现者只负责将这个指针转交给回调函数,而不关心它到底指向什么数据类型。调用者知道自己传的参数是char *型的,那么在自己提供的回调函数中就应该知道参数要转换成char *型来解释。  
  60. 回调函数的一个典型应用就是实现类似C++的泛型算法(Generics Algorithm)。下面实现的max函数可以在任意一组对象中找出最大值,可以是一组int、一组char或者一组结构体,但是实现者并不知道怎样去比较两个对象的大小,调用者需要提供一个做比较操作的回调函数。  
  61. 例 24.8. 泛型算法  
  62. /* generics.h */  
  63. #ifndef GENERICS_H  
  64. #define GENERICS_H  
  65.   
  66. typedef int (*cmp_t)(void *, void *);  
  67. extern void *max(void *data[], int num, cmp_t cmp);  
  68.   
  69. #endif  
  70. /* generics.c */  
  71. #include "generics.h"  
  72.   
  73. void *max(void *data[], int num, cmp_t cmp)  
  74. {  
  75.      int i;  
  76.      void *temp = data[0];  
  77.      for(i=1; i<num; i++) {  
  78.           if(cmp(temp, data[i])<0)  
  79.                temp = data[i];  
  80.      }  
  81.      return temp;  
  82. }  
  83. /* main.c */  
  84. #include <stdio.h>  
  85. #include "generics.h"  
  86.   
  87. typedef struct {  
  88.      const char *name;  
  89.      int score;  
  90. } student_t;  
  91.   
  92. int cmp_student(void *a, void *b)  
  93. {  
  94.      if(((student_t *)a)->score > ((student_t *)b)->score)  
  95.           return 1;  
  96.      else if(((student_t *)a)->score == ((student_t *)b)->score)  
  97.           return 0;  
  98.      else  
  99.           return -1;  
  100. }  
  101.   
  102. int main(void)  
  103. {  
  104.      student_t list[4] = {{"Tom", 68}, {"Jerry", 72},  
  105.                        {"Moby", 60}, {"Kirby", 89}};  
  106.      student_t *plist[4] = {&list[0], &list[1], &list[2], &list[3]};  
  107.      student_t *pmax = max((void **)plist, 4, cmp_student);  
  108.      printf("%s gets the highest score %d\n", pmax->name, pmax->score);  
  109.   
  110.      return 0;  
  111. }  
  112.   
  113. max函数之所以能对一组任意类型的对象进行操作,关键在于传给max的是指向对象的指针所构成的数组,而不是对象本身所构成的数组,这样max不必关心对象到底是什么类型,只需转给比较函数cmp,然后根据比较结果做相应操作即可,cmp是调用者提供的回调函数,调用者当然知道对象是什么类型以及如何比较。  
  114. 以上举例的回调函数是被同步调用的,调用者调用max函数,max函数则调用cmp函数,相当于调用者间接调了自己提供的回调函数。在实际系统中,异步调用也是回调函数的一种典型用法,调用者首先将回调函数传给实现者,实现者记住这个函数,这称为注册一个回调函数,然后当某个事件发生时实现者再调用先前注册的函数,比如sigaction(2)注册一个信号处理函数,当信号产生时由系统调用该函数进行处理,再比如pthread_create(3)注册一个线程函数,当发生调度时系统切换到新注册的线程函数中运行,在GUI编程中异步回调函数更是有普遍的应用,例如为某个按钮注册一个回调函数,当用户点击按钮时调用它。  
  115. 以下是一个代码框架。  
  116. /* registry.h */  
  117. #ifndef REGISTRY_H  
  118. #define REGISTRY_H  
  119.   
  120. typedef void (*registry_t)(void);  
  121. extern void register_func(registry_t);  
  122.   
  123. #endif  
  124.   
  125. /* registry.c */  
  126. #include <unistd.h>  
  127. #include "registry.h"  
  128.   
  129. static registry_t func;  
  130.   
  131. void register_func(registry_t f)  
  132. {  
  133.      func = f;  
  134. }  
  135.   
  136. static void on_some_event(void)  
  137. {  
  138.      ...  
  139.      func();  
  140.      ...  
  141. }  
  142.   
  143. 既然参数可以是函数指针,返回值同样也可以是函数指针,因此可以有func()();这样的调用。返回函数的函数在C语言中很少见,在一些函数式编程语言(例如LISP)中则很常见,基本思想是把函数也当作一种数据来操作,输入、输出和参与运算,操作函数的函数称为高阶函数(High-order Function)。  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值