定义
简单来说,回调函数是由别人的函数执行时调用你实现的函数。
函数指针作为某个函数的参数。函数指针变量可以作为某个函数的参数来使用的,回调函数就是一个通过函数指针调用的函数。
以下是来自知乎作者常溪玲的解说:
你到一个商店买东西,刚好你要的东西没有货,于是你在店员那里留下了你的电话,过了几天店里有货了,店员就打了你的电话,然后你接到电话后就到店里去取了货。在这个例子里,你的电话号码就叫回调函数,你把电话留给店员就叫登记回调函数,店里后来有货了叫做触发了回调关联的事件,店员给你打电话叫做调用回调函数,你到店里去取货叫做响应回调事件。
关键词汇
- 在这里,有4个关键的要素,分别是:登记、触发、调用、响应。
- 其实登记我理解就是注册,也就是把你的电话号码告诉店员的过程,你总得把你的电话放在店员那里吧,不然她怎么给你打电话,那么这个动作就是注册。
- 触发的过程,怎么说呢,就用触发这个词比较恰当,就像上面描述的,我知道店里来货了,这么个事情,其实就出发了我准备下一部去取货的动作(即调用)。
- 调用,其实就是店员给你打电话的过程,对应到代码中就是函数调用。
- 响应很简单,就是去店里拿货。
示例
那么,对应代码来看看:
实例中 populate_array 函数定义了三个参数,其中第三个参数是函数的指针,通过该函数来设置数组的值。实例中我们定义了回调函数 getNextRandomValue,它返回一个随机值,它作为一个函数指针传递给 populate_array 函数。populate_array 将调用 10 次回调函数,并将回调函数的返回值赋值给数组。
代码:
---------------------------------------------------------------------------------------------
PS:
1.登记在这里没写出来,应该是类似一个reg func(*callback(void))这样的注册动作。
2.触发也没有写出来,触发应该是触发了某一个事件、或者条件,比如:
if (某条件达到) {
//callback xxx
}
这里的“某条件达到”应该就是触发,当然,并不一定是if条件,在复杂的系统中,也可能是信号量等等。
3.调用就是下面的:populate_array(myarray, 10, getNextRandomValue);
4.响应其实就是对回调函数内部的处理了:
void populate_array(int *array, size_t arraySize, int (*getNextValue)(void))
{
for (size_t i=0; i<arraySize; i++)
array[i] = getNextValue();
}
---------------------------------------------------------------------------------------------
#include <stdlib.h>
#include <stdio.h>
// 回调函数
void populate_array(int *array, size_t arraySize, int (*getNextValue)(void))
{
for (size_t i=0; i<arraySize; i++)
array[i] = getNextValue();
}
// 获取随机值
int getNextRandomValue(void)
{
return rand();
}
int main(void)
{
int myarray[10];
populate_array(myarray, 10, getNextRandomValue);
for(int i = 0; i < 10; i++) {
printf("%d ", myarray[i]);
}
printf("\n");
return 0;
}