函数指针凭借其绚丽的外表迷惑了同学们很久,这一篇文章力争让同学们彻底明白函数指针的用法。
首先有两点是必须明白的,不明白也得先记住。
- 函数指针的形态
- C/C++ 标示符类型的定义规则
声明函数指针的形态如下
char (*f_ptr)(int, double);
//返回类型 (*标示符)(形参列表)
也就是说,与其他普通类型声明方法不同,函数指针的
声明并不是 [ 类型 标示符 ](如 int a)的形式。
这样
声明表达的意思是: f_ptr 是一个函数指针,指向一个函数的地址。
C/C++ 标示符的声明规则又是什么呢,通俗的讲:
对于一个声明标示符的语句,去掉分号与标示符本身,剩下的就是此标示符的类型。
按照这句话处理面函数指针的声明语句,就得到了
char (*)(int,double)
这就是 f_ptr 的类型
回忆 f_ptr 的调用方法
- 取解,获得该地址对应的函数
- 传参调用
由于括号 ( ) 的优先级高于 * ,因此取解必须加括号
char res = (*f_ptr)(1, 2.5);
下面通过三个例子来进一步说明函数指针的用法
void func1(int k){
printf("%d in func 1\n",k);
}
int main(){
/**
直接法:声明并初始化一个函数指针,然后调用
*/
void (*fp)(int) = &func1;
(*fp)(2);
/**
使用类型定义简化声明
将类型标识 fp_type 定义为 void (*)(int),然后定义函数指针
*/
typedef void(*fp_type)(int);
fp_type fp2 = &func1;
(*fp2)(3);
/**
类型转换法:
将一个 void * 指针 转化为函数指针的类型
使用 (转化类型)转换对象 格式
*/
void* fp3 = &func1;
(*(void(*)(int))fp3)(4);
system("pause");
}
怎么样,是不是觉得会使用函数指针了?明白了类型与标示符的关系,下次笔试让你写一个参数为函数指针的函数指针数组就不用怕了。
void (*b[10]) (void (*)());
函数指针有趣的例子还很多,在
这篇博客 看到一个。问:signal 是什么含义
void (*signal(int,void(*)(int)))(int);
在这里,signal不是函数指针,就是一个函数,其返回值是一个函数指针,怎么分析呢?首先看最外层 void (* BLABLA) (int),这就说明 BLABLA 是个函数指针
而 BLABLA 是 signal(int, 指针),很明显 “标识符(类型)”是函数签名的样子,而代表的含义正是这个函数的返回值。
欢迎指正!