1. 指针函数:指函数的返回值类型是一个指针类型,本质是一个函数,如int *function(void);返回值是int *,是一个指针;
2. 函数指针:本质是一个指针,但是这个指针指向一个函数,这个指针就叫函数指针。可把函数指针传入另一个函数作为形参,实现回调。
函数指针用途:调用函数;做函数的参数。
3. 回调函数:参数是一个函数指针,调用者可以传递一个函数的地址给实现者,让实现者去调用。
例:回调函数:void func(void(*f)(void *), void *p);
调用者:
提供一个回调函数,再提供一个准备传给回调函数的参数;
把回调函数传给参数f,把准备传给回调函数的参数按void *类型传给参数p。
实现者:
在适当的时候根据调用者传来的函数指针f调用回调函数,将调用者传来的参数p转交给回调函数,即调用f(p);
para_callback.h
#ifndef PARA_CALLBACK_H
#define PARA_CALLBACK_H
typedef void (*callback_t)(void *);
extern void repeat_three_times(callback_t, void *);
#endif
para_callback.c
#include "para_callback.h"
void repeat_three_times(callback_t f, void *para)
{
f(para);
f(para);
f(para);
}
main.c
#include <stdio.h>
#include "para_callback.h"
void say_hello(void *str)
{
printf("Hello %s\n", (const char *)str);
}
void count_numbers(void *num)
{
int i;
for(i=1; i<=(int)num; i++)
printf("%d ", i);
putchar('\n');
}
void (*pFun) (void *);
int main(void)
{
pFun = say_hello;// pFun的赋值可以不同,但一定要是函数的地址,并且署名和返回类型相同。
repeat_three_times(pFun, "Guys");
pFun = count_numbers;//给pFun赋不同的值(不同的函数地址),则调用者将调用不同地址的函数。赋值可发生在运行时,从而实现动态绑定。
repeat_three_times(pFun, (void *)4);
return 0;
}
上例表明:回调函数的参数按什么类型解释由调用者规定,对于实现者来说就是一个void *指针,实现者只负责将这个指针转交给回调函数,而不关心它到底指向什么数据类型。调用者知道自己传的参数是char*型的,那么在自己提供的回调函数中就应该知道参数要转换成char*型解释。
以上举例的回调函数是被同步调用的,调用者调用repeat_three_times函数,repeat_three_times函数则调用say_hello或count_numbers函数,相当于调用者间接调了自己提供的回调函数。在实际系统中,异步调用也是回调函数的一种典型用法,调用者首先将回调函数传给实现者,实现者记住这个函数,这称为注册一个回调函数,然后当某个事件发生时实现者再调用先前注册的函数.