回调函数定义
回调函数
指在程序执行期间通过传递函数指针(地址)来调用的一段用户实现的函数。在C语言中,回调函数通常作为函数参数传递给另一个调用方函数,并由该调用方函数在适当的时候调用(比如当某个条件满足时)。回调函数的主要用途是实现更高级别的控制结构以及异步事件处理。例如,在图形用户界面中,当用户点击一个按钮时,会执行一个回调函数来处理该事件。此外,回调函数还可以在处理大量数据或循环过程中提供有效的处理方法。
回调函数作用
因为可以把调用者与被调用者分开。调用者不关心谁是被调用者,所有它需知道的,只是存在一个具有某种特定原型、某些限制条件(如返回值为int)的被调用函数。
如果想知道回调函数在实际中有什么作用,先假设有这样一种情况,我们要编写一个库,它提供了某些排序算法的实现,如冒泡排序、快速排序、shell排序、shake排序等等,但为使库更加通用,不想在函数中嵌入排序逻辑,而让使用者来实现相应的逻辑;或者,想让库可用于多种数据类型(int、float、string),此时,该怎么办呢?可以使用函数指针,并进行回调。
回调可用于通知机制,例如,有时要在程序中设置一个计时器,每到一定时间,程序会得到相应的通知,但通知机制的实现者对我们的程序一无所知。而此时,就需有一个特定原型的函数指针,用这个指针来进行回调,来通知我们的程序事件已经发生。实际上,SetTimer()API使用了一个回调函数来通知计时器,而且,万一没有提供回调函数,它还会把一个消息发往程序的消息队列。
另一个使用回调机制的API函数是EnumWindow(),它枚举屏幕上所有的顶层窗口,为每个窗口调用一个程序提供的函数,并传递窗口的处理程序。如果被调用者返回一个值,就继续进行迭代,否则,退出。EnumWindow()并不关心被调用者在何处,也不关心被调用者用它传递的处理程序做了什么,它只关心返回值,因为基于返回值,它将继续执行或退出。
不管怎么说,回调函数是继续自C语言的,因而,在C++中,应只在与C代码建立接口,或与已有的回调接口打交道时,才使用回调函数。除了上述情况,在C++中应使用虚拟方法或函数符(functor),而不是回调函数。
回调函数举例
当我们需要将某个函数作为参数传递到另一个函数中并在另一个函数中执行时,通常可以使用回调函数。在 C 语言中,常见的回调函数举例如下:
1、打包成lib的常见形式
//消费者
//回调函数1
//生成一个2k形式的偶数
#include <stdio.h>
int double_1(int x)/*参考回调函数声明*/
{
return x * 2;
}
//回调函数2
//生成一个4k形式的偶数
int double_2(int x)
{
return x * 4;
}
//生产者
//中间函数
//接受一个生成偶数的函数作为参数
//返回一个奇数
int (*fun)(int data);/*回调函数声明*/
int getOddNumber(int k, int (*fun)(int))
{
return 1 + fun(k);
}
//消费者
//起始函数,这里是程序的主函数
int main()
{
int k = 1,ret = 0;
k = 1;
//当需要生成一个2k+1形式的奇数时
ret = getOddNumber(k,double_1);
//当需要一个4k+1形式的奇数时
ret = getOddNumber(k,double_2);
}
2、实际项目中可以参考:
编写和使用回调函数的流程:
1、自定义函数指针,形参作为硬件层要传到应用层的数据。
2、硬件层定义一个函数指针和函数指针注册函数。
3、应用层定义一个函数,返回值和形参都要和函数指针一致。
4、应用层调用函数指针注册函数,把定义好的函数名称作为形参传入。
消费者
生产者
生产者声明:
生产者:
3、函数指针作为转换表
#include<iostream>
#include<cmath>
using namespace std;
// 转换表
// 转换表 step1:
//(1.1)声明 转台转移函数
double add(double, double);
double sub(double, double);
double mul(double, double);
double div(double, double);
double xiebianchang(double, double);
//(1.2)声明并初始化一个函数指针数组 pfunc:数组 数组元素:函数指针 返回值:double型数据
double(*pfunc[])(double, double) = { add, sub, mul, div, xiebianchang };//5个转移状态
//状态转移函数的实现
double add(double a, double b){ return a + b;}
double sub(double a, double b){ return a - b; }
double mul(double a, double b){ return a*b; }
double div(double a, double b){ return a / b; }
double xiebianchang(double a, double b){ return sqrt(pow(a, 2) + pow(b, 2)); }
void test(){
//转换表 step2:调用 函数指针数组
int n = sizeof(pfunc) / sizeof(pfunc[0]);//转移表中 包含的元素个数(状态转移函数个数)
for (int i = 0; i < n; ++i){
cout << pfunc[i](3, 4) << endl;
}
}
int main(){
test();
return 0;
}