回调函数
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
回调函数是一个程序员不能显式调用的函数;通过将回调函数的地址传给被调用者从而实现调用。
回调函数实现的机制是:
(1)定义一个回调函数;
(2)提供函数实现的一方在初始化的时候,将回调函数的函数指针注册给调用者;
(3)当特定的事件或条件发生的时候,调用者使用函数指针调用回调函数对事件进行处理。
它是一个由调用方自己实现,供被调用方使用的特殊函数。
被调用者回头调用调用者的函数(够咬嘴的),故称其为回调(callback)。
精妙比喻:回调函数还真有点像您随身带的BP机:告诉别人号码,在它有事情时Call您。
回调用于层间协作,上层将本层函数安装在下层,这个函数就是回调,而下层在一定条件下触发回调,例如作为一个驱动,是一个底层,他在收到一个数据时,除了完成本层的处理工作外,还将进行回调,将这个数据交给上层应用层来做进一步处理,这在分层的数据通信中很普遍。其实回调和API非常接近,他们的共性都是跨层调用的函数。但区别是API是低层提供给高层的调用,一般这个函数对高层都是已知的;而回调正好相反,他是高层提供给底层的调用,对于低层他是未知的,必须由高层进行安装,这个安装函数其实就是一个为低层提供的API,安装后低层不知道这个回调的名字,但它通过一个函数指针来保存这个回调,在需要调用时,只需引用这个函数指针和相关的参数指针。其实:回调就是该函数写在高层,低层通过一个函数指针保存这个函数,在某个事件的触发下,低层通过该函数指针调用高层那个函数。
为什么要使用回调函数?
因为可以把调用者与被调用者分开。调用者不关心谁是被调用者,所有它需知道的,只是存在一个具有某种特定原型、某些限制条件(如返回值为int)的被调用函数。
如果想知道回调函数在实际中有什么作用,先假设有这样一种情况,我们要编写一个库,它提供了某些排序算法的实现,如冒泡排序、快速排序、shell排序、shake排序等等,但为使库更加通用,不想在函数中嵌入排序逻辑,而让使用者来实现相应的逻辑;或者,想让库可用于多种数据类型(int、float、string),此时,该怎么办呢?可以使用函数指针,并进行回调。
应用举例
C语言的标准库函数中很多地方就采用了回调函数来让用户定制处理过程。如常用的快速排序函数、二分搜索函数等。
qsort是一个快排序的函数。
需要调用一个用户自定义的比较函数。
stdlib.h
void qsort(void *base, size_t nelem, size_t width, int(*fcmp)(const void *, const void *) );
base 待排序的数组,melem数组的元素个数,width每个元素的大小,fcmp用于对数组元素进行比较的函数指针,该函数由自己另外编写,有2个参数。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int sort_function_for_int( const void *a, const void *b);
int sort_function_for_char( const void *a, const void *b);
int sort_function_for_string( const void *a, const void *b);
int main(int argc, char **argv)
{
int i;
int list[5] = { 54, 21, 11, 67, 22 };
char list1[5] = {'d', 'a', 'z', 'b', 'q'};
const char* list2[]={"estella", "danielle", "crissy", "bo", "angie"};
qsort((void *)list, 5, sizeof(list[0]), sort_function_for_int);
for (i = 0; i < 5; i++)
printf("%d\n", list[i]);
qsort((void *)list1, 5, sizeof(list1[0]), sort_function_for_char);
for (i = 0; i < 5; i++)
printf("%c\n", list1[i]);
qsort((void *)list2, 5, sizeof(char *), sort_function_for_string);
for (i = 0; i < 5; i++)
printf("%s\n", list2[i]);
return 0;
}
int sort_function_for_int( const void *a, const void *b)
{
return *(int*)a-*(int*)b;
}
int sort_function_for_char(const void *a, const void *b)
{
return *(char *)a - *(char *)b;
}
int sort_function_for_string( const void *a, const void *b)
{
return strcmp((char *)a, (char *)b);
}