(八)函数指针
1.函数的地址
与变量一样,函数也是有地址的,函数的地址指的是存储函数的物理内存的地址,可以编写以函数地址为参数的函数,一般说来这样比直接使用函数要笨拙,但是这可以使函数在不同的时间传递不同函数的地址,也就是使用不同的函数,在某些情况下是有用的。
2.函数指针的相关知识
(1)要使用函数指针来调用函数,需要:获取函数的地址;声明一个函数指针;使用函数指针来调用函数。
(2)获取函数的地址。只要函数后面不加参数,那么函数名就是函数的地址。比如think()是一个函数,那么think就是函数的地址。一定要区分传递的是函数的地址还是函数的返回值,函数的地址是函数名,不带括号和参数,而如果带了括弧和参数,则将会调用函数,而将函数的返回值传递过来。
(3)声明函数指针。int (*a)(int b);或者直接int (*a)(int);其中函数的返回值和参数列表(特征标)都要跟原函数的声明一模一样。一般我们可以在声明完了函数之后,将函数名改写成指针形式(*a)即可声明指针。这里要注意括号不能省略,省略了括号就是声明了一个返回指针的函数,而不省略括号就是声明了一个函数指针。
(4)使用指针来调用函数。这有两种方式,都是可以的,我们一般使用第一种。第一种是使用*a表示函数名,*a就和函数名一样,用法也是相同的。第二种方法是直接使用指针,也跟函数名一样的用法。这是两种理念,是矛盾的,但在c++中,选择了两种方式都允许的策略。
3.深入探讨函数指针
可以在声明的时候对函数指针进行初始化,比如int(*a)(double)=p;p是一个参数和返回值跟*a相同的函数名。也可以使用自动匹配的auto的形式声明指针,比如auto a=p;p是一个函数名,这是一种更简单的方式。
如果要声明函数指针数组,应该在什么地方加[]呢?答案如下:int (*a[3])(double)={f1,f2,f3};[]的优先级要比*高,因此这里是声明了包含有三个指针的函数指针数组。自动类型匹配只适用于单值初始化,不能用于初始化列表。
当函数指针和数组等知识结合起来使用的时候,将会非常复杂,但是只要注意一点就可以顺利破解难题,那就是数组名实际上是数组第一个元素的地址,而对数组名取地址(如&a)表示的是整个数组的地址或者说是一个二维数组名,需要进行两次解除引用才能得到第一个元素的内容。知道了这一点后,即使加上函数指针的内容,也不难理解。
函数名与函数地址的作用相同。
!!!!!使用函数指针,可以实现由一种接口实现不同的功能,对于程序的模块化设计非常有利。
4.使用typedef进行简化
除了使用类型自动推断auto来进行声明简化之外,还可以利用typedef来进行声明的简化,typedef的用法是声明一个类型的别名,前面加上typedef,后面相当于声明了一个变量,不同的是这个变量名就是类型的别名。比如typedef const double *(*p_function)(const double*,int);p_function p1=f1;p_function就是一个函数指针的别名,可以用这个别名对函数指针声明进行简化。