表 24.7. 回调函数示例:void func(void (*f)(void *), void *p);

    例 24.7. 回调函数

    1. /* para_callback.c */
    2. #include "para_callback.h"
    3.  
    4. void repeat_three_times(callback_t f, void *para)
    5. {
    6. f(para);
    7. f(para);
    8. f(para);
    9. }

    回顾一下前面几节的例子,参数类型都是由实现者规定的。而本例中回调函数的参数按什么类型解释由调用者规定,对于实现者来说就是一个void *指针,实现者只负责将这个指针转交给回调函数,而不关心它到底指向什么数据类型。调用者知道自己传的参数是char *型的,那么在自己提供的回调函数中就应该知道参数要转换成char *型来解释。

    回调函数的一个典型应用就是实现类似C++的泛型算法(Generics Algorithm)。下面实现的max函数可以在任意一组对象中找出最大值,可以是一组int、一组char或者一组结构体,但是实现者并不知道怎样去比较两个对象的大小,调用者需要提供一个做比较操作的回调函数。

    例 24.8. 泛型算法

    1. /* generics.h */
    2. #ifndef GENERICS_H
    3. #define GENERICS_H
    4.  
    5. typedef int (*cmp_t)(void *, void *);
    6. extern void *max(void *data[], int num, cmp_t cmp);
    7.  
    8. #endif
    1. #include <stdio.h>
    2. #include "generics.h"
    3.  
    4. typedef struct {
    5. const char *name;
    6. int score;
    7. } student_t;
    8.  
    9. int cmp_student(void *a, void *b)
    10. {
    11. if(((student_t *)a)->score > ((student_t *)b)->score)
    12. return 1;
    13. else if(((student_t *)a)->score == ((student_t *)b)->score)
    14. return 0;
    15. else
    16. return -1;
    17. }
    18.  
    19. int main(void)
    20. {
    21. student_t list[4] = {{"Tom", 68}, {"Jerry", 72},
    22. student_t *plist[4] = {&list[0], &list[1], &list[2], &list[3]};
    23. student_t *pmax = max((void **)plist, 4, cmp_student);
    24. printf("%s gets the highest score %d\n", pmax->name, pmax->score);
    25.  
    26. return 0;
    27. }

    以上举例的回调函数是被同步调用的,调用者调用max函数,max函数则调用cmp函数,相当于调用者间接调了自己提供的回调函数。在实际系统中,异步调用也是回调函数的一种典型用法,调用者首先将回调函数传给实现者,实现者记住这个函数,这称为注册一个回调函数,然后当某个事件发生时实现者再调用先前注册的函数,比如sigaction(2)注册一个信号处理函数,当信号产生时由系统调用该函数进行处理,再比如pthread_create(3)注册一个线程函数,当发生调度时系统切换到新注册的线程函数中运行,在GUI编程中异步回调函数更是有普遍的应用,例如为某个按钮注册一个回调函数,当用户点击按钮时调用它。

    以下是一个代码框架。

    1. /* registry.c */
    2. #include <unistd.h>
    3. #include "registry.h"
    4.  
    5. static registry_t func;
    6.  
    7. void register_func(registry_t f)
    8. {
    9. func = f;
    10. }
    11.  
    12. static void on_some_event(void)
    13. {
    14. ...
    15. func();
    16. }

    既然参数可以是函数指针,返回值同样也可以是函数指针,因此可以有func()();这样的调用。返回函数的函数在C语言中很少见,在一些函数式编程语言(例如LISP)中则很常见,基本思想是把函数也当作一种数据来操作,输入、输出和参与运算,操作函数的函数称为高阶函数(High-order Function)。

    1、的5.6节有一个函数的实现,可以对一组任意类型的对象做快速排序。请读者仿照那个例子,写一个插入排序的函数和一个折半查找的函数。