函数指针的两种调用形式
https://www.zhihu.com/question/421214887
1 void test(int); 2 int main(void){ 3 void (*fp)(int); 4 fp=test; 5 (*fp)(9); 6 7 fp(9); 8 return 0; 9 } 10 void test(int a) 11 { 12 printf( "%d\n", a ); 13 } 14 15 16 这两种形式是等价的,都是为ANSI C所接受的。 17 历史上,贝尔实验室的C和UNIX的开发者采用的是第一种观点 18 QUOTE: 19 (*fp)(9); 20 而Berkeley的UNIX的扩展者采用第二种观点 21 QUOTE: 22 fp(9); 23 24 25 K&R C不允许第二种形式。 26 但是为了保持与现有代码的兼容性,ANSI C把这两者作为等价形式全部接受。 27 28 29 正如 kernelxu 所言,按照标准 C 的规定两种函数调用方式都是正确的。 30 这是因为在 C/C++ 中总是使用函数指针的形式来调用函数。即使在函数调用中使用的是函数指示符(代表函数类型), 31 也会被转换为函数指针使用,这就是默认的 function-to-pointer 转换。 32 33 例如,楼主程序中的 test 函数可以直接使用函数指示符形式来调用:test( 9 );。 34 然而,在这里的函数指示符 test 其实被编译器自动转换为了函数指针来使用,即从函数类型转换为了函数指针类型, 35 最终是使用函数指针的形式来完成对函数调用的。 36 37 程序中的 fp( 9 ); 是直接使用函数指针 fp 来调用。既然 fp 已经是函数指针了,所以在类型上就不需要任何转换了。 38 (*fp)( 9 ); 也是合法的函数调用。在这里,fp 是函数指针,所以 *fp 是对于函数的引用,是函数类型。 39 根据标准规定的 function-to-pointer 转换又把 *fp 由函数类型转换为了函数指针类型,所以实际上 (*fp)( 9 ); 40 相当于 fp( 9 ); 这种直接的函数指针调用方式。 41 42 另外,test 函数也可这样调用:(*test)( 9 );。可以这样来理解:根据 function-to-pointer 转换规定 43 首先 test 由函数类型转换为函数指针,那么 *test 表示的又是函数类型, 44 最后又根据 function-to-pointer 转换为函数指针来调用函数。这其实和 (*fp)( 9 ); 是等价的。 45 甚至 test 函数还可以这样调用:(**test)( 9 );、(***test)( 9 );、(****test)( 9 ); 等等, 46 或者函数指针形式:(**fp)( 9 );、(***fp)( 9 );、(****fp)( 9 ); 等等。对此的理解可参看上段中的分析。 47 从上面的分析可以看出,函数调用的时候可以使用函数指针的方式,也可以使用函数指示符的方式。 48 不过,后者会由编译器自动转换为前者的形式,即函数指针的形式。和指向对象的指针相比,这是函数指针一个比较特殊的地方。