在C语言中
函数本身不是变量
但可以定义指向函数的指针
这种指针可以被赋值
存放于数组之中
传递给函数及作为函数的返回值等
--《The C Programming Language Second Edition》
下面给出几个简单的例子来说明指向函数的指针
第一个例子说明指向函数的指针如何说明赋值调用
#include #define TESTDATE
int func(int a)/*func用于打印一个整数*/
{
return printf(%d\na);
}
main()
{
int (*FunctionPionter)(int a);
FunctionPionter = func;
(*FunctionPionter)(TESTDATE);
return ;
}
其中重点语句的含义如下
int (*FunctionPionter)(int a);
FunctionPionter: 指向一个返回整数的函数的指针这个指针有一个整数参数
FunctionPionter = func;
将FunctionPionter指向函数func其中函数必须已经定义且函数和函数指针的说明的返回值必须一致
(*FunctionPionter)(TESTDATE);
通过函数指针调用函数因为函数指针已经指向函数所以用*取出函数指针的内容就为函数本身
下面这个例子显示如何将指向函数的指针传递给函数作为函数的返回类型在这个例子中有三个函数
hello返回字符指针的函数用来返回字符串hello world!\n
RetFunc返回一个指向函数的指针的函数且返回指针所指的那个函数为一个返回字符指针的函数
call返回一个void *型的指针且call有一个指向函数的指针的参数且这个函数指针返回一个字符指针
#include #define MAX
main()
{
void *call(char *(*)());
char *(*RtnFunc())();
/*上面两个说明有些复杂*/
printf(%scall(RtnFunc()));
return ;
}
char *hello()
{
return Hello World!\n;
}
char *(*RtnFunc())()
{
return hello;
}
void *call(char *(*func)())
{
return (*func)();
}
上面的例子中main()无法直接调用hello函数利用两个函数分别返回hello和调用hello实现了在main()中调用hello虽然似乎这个程序显得多余但却很好的说明了如何把指向函数的指针传递给函数作为函数的返回其中call函数利用了void *型指针的灵活机制使得call的适用性大为增加这也正是指向函数的指针的优点之一同样的例子是《The C Programming Language Second Edition》中下面这个函数调用
qsort((void **) lineptr nlines (int (*)(void * void *))(numeric ? numcmp : strcmp));
其中使用了两次强制类型转换其中第二甚至是利用指向函数的指针将函数的类型进行了转换当然上面语句在某些编译器上无法通过因为某些编译器要求条件表达
表达式 ? 表达式 : 表达式
中表达式2与表达式3的类型相同当然这样的要求是不符合ANSI标准的在ANSI标准中如果表达式2与表达式3的类型不同则结果的类型由类型转换规则决定当然我们可以变同一下先将两个函数的类型进行强制转换来达到目的
qsort((void **) lineptr nlines numeric ? (int (*)(void * void *))numcmp : (int (*)(void * void *))strcmp));
对于如何直接说明一个像RtnFunc一样返回指向函数的指针的函数我查阅了不少资料都没有找到答案最后是自己硬着头皮摸索出来的由此我也对C的复杂说明有了更深刻的体会将在以后的技术日记中写出来当然在我看来过多的不合适的使用这些复杂说明并不是一种好的编程风格因为它将使程序变得难以理解同时也增加了出错的可能性
一个比较好的折衷的方法是使用typedef来使程序的含义明朗下面给出用typedef给写上面那个程序的例子其中定义个一个类型PtoFun用typedef说明PtoFun是指向函数的指针类型指针所指的函数返回一个字符指针且没有参数
#include #define MAX
typedef char *(*PtoFun)();
main()
{
void *call(PtoFun);
PtoFun RtnFunc();
printf(%scall(RtnFunc()));
return ;
}
char *hello()
{
return Hello World!\n;
}
PtoFun RtnFunc()
{
return hello;
}
void *call(PtoFun func)
{
return (*func)();
}
改写后的程序的可读性大为增加给人一目了然的感觉