指针函数 vs 函数指针
函数指针和指针函数是C语言中的一个挺常见的问题,对于新手来说,若不加以区分,还真是个难点。针对此问题,本章专门介绍了二者之间的区别,让读者对函数指针和指针函数有更深入的了解。
函数指针和指针函数都是概念简称,指针函数是指返回值是指针的函数,即本质是一个函数。而函数指针是指向函数的指针变量,即本质是一个指针变量。
(1)指针函数
函数指针是指返回值是指针的函数,其本质是一个函数。函数都是返回类型,只不过指针函数返回类型是某一类型的指针。其定义格式为:
返回类型 * 函数名(参数列表)
返回类型可以是任何基本类型和复合类型。事实上,每一个函数,即使它不带有返回某种类型的指针,它本身都有一个入口地址,该地址相当于一个指针。比如函数返回一个整型值,实际上也相当于返回一个指针变量的值,不过这时的变量是函数本身而已,而整个函数相当于一个“变量”。
函数指针的实例:
#include <iostream>
using namespace std;
int * max(int a,int b,int *d)
{
*d=(a>b?a:b);
return d;
}
int main(int argc,char *argv)
{
int a=10;
int b=9;
int d;
int *c=max(a,b,&d);
cout<<*c<<endl;
return 0;
}
这只是一个非常简单的例子,是为了单纯练习指针函数而写的。指针函数最重要的一点就是:返回的指针一定不能是函数局部对象的指针,即,函数调用完成后,该指针指向的地址的生命周期仍没有结束。例子中指针指向的是d的地址,所以函数调用完成后d仍是有效的。若返回的是函数中定的变量的地址,则会出错。
p.s. 注意函数的返回值:无返回值、返回普通类型或自定义类型(复制一个临时变量)、返回引用(返回对象本身)、返回指针,尤其是返回引用和于返回指针的区别,关于返回引用,可参考:http://dong008259.blog.163.com/blog/static/51645417201110394255378/
(2)函数指针
函数指针是指一个函数类型的指针变量,实质是一个指针变量。这正如用指针变量可指向整型变量、字符型、数组一样,这里是指向函数。如前所述,C在编译时,每一个函数都有一个入口地址,该入口地址就是函数指针所指向的地址。有了指向函数的指针变量后,可用该指针变量调用函数,就如同用指针变量可引用其他类型变量一样,其定义格式为:
返回类型 (*指针变量名) (参数列表)
如: int func(int x); /* 声明一个函数 */
int (*f) (int x); /* 声明一个函数指针 */
f=func; /* 将func函数的首地址赋给指针f */
函数指针实例:
#include <iostream>
using namespace std;
int max(int a,int b)
{
return (a>b?a:b);
}
int main(int argc,char *argv)
{
int (*ptr)(int,int);
ptr=max;
int a=10;
int b=9;
int c=(*ptr)(a,b);
cout<<c<<endl;
return 0;
}
ptr是指向函数的指针变量,所以可把函数max()赋给ptr作为ptr的值,即把max()的入口地址赋给ptr,以后就可以用ptr来调用该函数,实际上ptr和max都指向同一个入口地址,不同就是ptr是一个指针变量,不像函数名称那样是死的,它可以指向任何函数,就看你想怎么做了。在程序中把哪个函数的地址赋给它,它就指向哪个函数。而后用指针变量调用它,因此可以先后指向不同的函数,所以函数指针具有很高的灵活性,赋给函数指针的函数应该和函数指针所指的函数原型是一致的。不过注意,指向函数的指针变量没有++和--运算,用时要小心。也可进行宏定义:typedef int (*fun_ptr)(int,int);
关于函数指针相信大家都理解地差不多了,以前遇到这么一个题目,“(*((void(*)(void))0x12345678))();”这个表达式应如何理解?
个人理解:void (*)(void)是一个函数指针;
(void (*)(void)0x12345678)是将0x12345678强制转换为函数指针;
*(void (*)(void)0x12345678)是取指针所指向地址的内容,即一个函数;
(*(void (*)(void)0x12345678))();是调用这个函数。