#include "stdio.h"
#include "stdlib.h"int add(int a,int b)
{ return a+b;}int sub(int a,int b)
{ return a-b;}typedef int (*fcb_fun)(int a,int b);
typedef struct mydata
{ int a,b; fcb_fun fcb;}mydata_t;mydata_t *mydata_create(fcb_fun out_fcb)
{ mydata_t *data = (mydata_t *)malloc(sizeof(mydata_t)); data->fcb = out_fcb; data->a = 0; data->b = 0;return data;
}void mydata_release(mydata_t *data)
{ free(data);}void mydata_operate(mydata *data)
{ int c =data->fcb(data->a,data->b); printf("mydata inner operate %d \n",c);}int main(void)
{ mydata_t *data =mydata_create(add); data->a =10; data->b =3; mydata_operate(data);data->fcb = sub;
mydata_operate(data); mydata_release(data);return 0;
}
使用机制
⑴定义一个回调函数;
⑵提供函数实现的一方在初始化的时候,将回调函数的函数 注册给调用者;
⑶当特定的事件或条件发生的时候,调用者使用 调用回调函数对事件进行处理。
因为可以把调用者与被调用者分开,所以调用者不关心谁是被调用者。它只需知道存在一个具有特定原型和限制条件的 。简而言之,回调函数就是允许用户把需要调用的方法的指针作为参数传递给一个函数,以便该函数在处理相似事件的时候可以灵活的使用不同的方法。
想知道回调函数在实际中有什么作用?先假设有这样一种情况:我们要编写一个库,它提供了某些 的实现(如 、 、 、shake排序等等),为了能让库更加通用,不想在函数中嵌入排序逻辑,而让使用者来实现相应的逻辑;或者,能让库可用于多种数据类型(int、float、string),此时,该怎么办呢?可以使用 ,并进行回调。
回调可用于通知机制。例如,有时要在A程序中设置一个 ,每到一定时间,A程序会得到相应的通知,但通知机制的实现者对A程序一无所知。那么,就需一个具有特定原型的函数指针进行回调,通知A程序事件已经发生。实际上, 使用一个回调函数SetTimer()来通知计时器。如果没有提供回调函数,它还会把一个消息发往程序的 。
另一个使用回调机制的API函数是EnumWindow(),它 屏幕上所有的顶层窗口,每个窗口都可以通过它调用另一个程序提供的函数,并传递窗口的处理程序。例如:如果被调用者返回一个值,就继续进行 ;否则,退出。EnumWindow()并不关心被调用者在何处,也不关心被调用者用它传递的处理程序做了什么,它只关心返回值,因为基于返回值,它将继续执行或退出。
不管怎么说,回调函数是继承自C语言的。在C++中,应只在与C代码建立接口或与已有的回调接口打交道时,才使用回调函数。除了上述情况,在C++中应使用虚拟方法或 (functor),而不是回调函数。