之前对于C/C++的回调函数不是很理解,今天就来学习下C/C++中的回调函数。
在理解回调函数之前,首先讨论下函数指针的概念。
函数指针:
1. 概念:指针是一个变量,是用来指向内存地址的。一个程序运行时,所有和运行相关的事物都是需要加载到内存中,这就决定了程序运行时的任何物件都可以用指针来指向他。函数是存放在内存代码区域内的,他们同样有地址,因此同样可以用指针来存取函数,把这个指向函数入口地址的指针称为函数指针。
2. 先看一个Hello World程序:
1 #include <stdio.h> 2 3 int main() 4 { 5 printf("Hello World!\n"); 6 return 0; 7 }
然后,采用函数调用的形式来实现:
1 #include <stdio.h> 2 3 void TestFunc(char *str) 4 { 5 printf("%s\n", str); 6 } 7 8 int main() 9 { 10 TestFunc("Hello World!"); 11 return 0; 12 }
用函数指针的方式来实现:
1 #include <stdio.h> 2 3 void TestFunc(char *str) 4 { 5 printf("%s\n", str); 6 } 7 8 int main() 9 { 10 void (*fp) (char *); // 声明一个函数指针(fp) 11 fp = TestFunc; // 将TestFunc函数的入口地址赋值给fp 12 fp("Hello World!"); // 函数指针fp实现函数调用 13 return 0; 14 }
由上可知:函数指针的声明和函数的声明之间唯一区别就是,用指针名(*fp)代替了函数名TestFunc。函数指针声明后,对函数指针进行赋值 fp = TestFunc就可以进行函数指针的调用了。声明函数指针时,只要函数返回值类型,参数个数,参数类型保存一致即可。注意,函数指针必须要用括号括起来:void (*fp) (char *) 或者 void (*fp) (char *str)
实际中,为了方便,通常用宏定义的方式来声明函数指针,例子程序如下:
1 #include <stdio.h> 2 3 typedef void (*FP) (char *); 4 5 void TestFunc(char *str) 6 { 7 printf("%s\n", str); 8 } 9 10 int main() 11 { 12 FP fp; // 通常是用宏FP来声明一个函数指针fp 13 fp = TestFunc; // 将TestFunc函数的入口地址赋值给fp 14 fp("Hello World!"); // 函数指针fp实现函数调用 15 return 0; 16 }
函数指针数组
同样用程序对函数指针数组进行探讨。
1 #include <stdio.h> 2 3 typedef void (*FP) (char *); 4 void f1(char *s) { printf(s); } 5 void f2(char *s) { printf(s); } 6 void f3(char *s) { printf(s); } 7 8 int main() 9 { 10 void* a[]={f1,f2,f3}; //定义了指针数组,这里a是一个普通指针 11 a[0]("Hello World!\n"); //编译错误,指针数组不能用下标的方式来调用函数 12 13 FP f[]={f1,f2,f3}; //定义一个函数指针的数组,这里的f是一个函数指针 14 f[0]("Hello World!\n"); //正确,函数指针的数组进行下标操作可以进行函数的间接调用 15 16 return 0; 17 }
回调函数
1. 概念:回调函数,顾名思义,就是使用者自己定义一个函数,使用者自己实现这个函数的程序内容,然后把这个函数作为参数传入别人(或系统)的函数中,由别人(或系统)的函数在运行时来调用的函数。函数是你实现的,但由别人(或系统)的函数在运行时通过参数传递的方式调用。这就是所谓的回调函数。简单来说,就是由别人的函数运行期间来回调你实现的函数。
2. 还是先从 Hello World程序看起。
1 #include <stdio.h> 2 3 int main() 4 { 5 printf("Hello World!\n"); 6 return 0; 7 }
将他修改成函数回调:
1 #include <stdio.h> 2 3 // 定义回调函数 4 void TestFunc() 5 { 6 printf("Hello World!\n"); 7 } 8 9 // 定义实现回调函数的“回调函数” 10 void CallTestFunc(void (* callfunc ) ()) 11 { 12 callfunc(); 13 } 14 15 // 在main函数中实现函数回调 16 int main() 17 { 18 CallTestFunc(TestFunc); 19 return 0; 20 }
再看看带参数的回调函数例子:
1 #include <stdio.h> 2 3 // 定义带参数回调函数 4 void TestFunc(char *str) 5 { 6 printf(str); 7 } 8 9 // 定义实现带参数回调函数的“回调函数” 10 void CallTestFunc(void (* callfunc ) (char *), char *str) 11 { 12 callfunc(str); 13 } 14 15 // 在main函数中实现带参数函数回调 16 int main() 17 { 18 CallTestFunc(TestFunc, "Hello World!"); 19 return 0; 20 }
至此,对函数指针和回调函数应该有个大致的了解了。