接上一篇:C语言函数指针总结(1)程序员常常需要实现回调。本文将讨论函数指针的基本原则并说明如何使用函数指针实现回调。注意这里针对的是普通的函数。
回调函数是一个程序员不能显式调用的函数,通过将回调函数的地址传给调用者从而实现调用。要实现回调,必须首先定义函数指针。尽管定义的语法有点不可思议,但如果你熟悉函数声明的一般方法,便会发现函数指针的声明与函数声明非常类似。 回调函数使用有时候是必要的,在我们想通过一个统一接口实现不同的内容,这时用回掉函数非常合适。比如,我们为几个不同的设备分别写了不同的显示函数:void TVshow(); void ComputerShow(); void NoteBookShow()...等 等。这是我们想用一个统一的显示函数,我们这时就可以用回调函数了。void show(void (*ptr)()); 使用时根据所传入的参数不同而调用不同的回调函数。 不同的编程语言可能有不同的语法。请看下面的例子:
#include <stdlib.h> #include <stdio.h> int Test1() { int i; for (i=0; i<30; i++) { printf("The %d th charactor is: %c\n", i, (char)('a' + i%26)); } return 0; } int Test2(int num) { int i; for (i=0; i<num; i++) { printf("The %d th charactor is: %c\n", i, (char)('a' + i%26)); } return 0; } void Caller1(void (*ptr)())//指向函数的指针作函数参数 { (*ptr)(); } void Caller2(int n, int (*ptr)())//指向函数的指针作函数参数,这里第一个参数是为指向函数的指针服务的 { //不能写成void Caller2(int (*ptr)(int n)),这样的定义语法错误。 (*ptr)(n); return; } int main() { printf("************************\n"); Caller1(Test1); //相当于调用Test2(); printf("&&&&&&************************\n"); Caller2(30, Test2); //相当于调用Test2(30); return 0; }
以上通过将回调函数的地址传给调用者从而实现调用,但是需要注意的是带参回调函数的用法。要实现回调,必须首先定义函数指针。
C语言的标准库函数中很多地方就采用了回调函数来让用户定制处理过程。如常用的快速排序函数、二分搜索函数等。快速排序函数原型:
void qsort(void *base, size_t nelem, size_t width, int (_USERENTRY *fcmp)(const void *, const void *)); 二分搜索函数原型: void *bsearch(const void *key, const void *base, size_t nelem, size_t width, int (_USERENTRY *fcmp)(const void *, const void *));
其中fcmp就是一个回调函数的变量。下面给出一个具体的例子:
#include <stdio.h> #include <stdlib.h> int sort_function( const void *a, const void *b); int list[5] = { 54, 21, 11, 67, 22 }; int main(void) { int x; qsort((void *)list, 5, sizeof(list[0]), sort_function); for (x = 0; x < 5; x++) printf("%i\n", list[x]); return 0; } int sort_function( const void *a, const void *b) { return *(int*)a-*(int*)b; }