这是一个群友发的笔试题目,里面涉及的东西也比较有意思。
直接看代码
void (*f[])(char *)
这个是个什么东西?
我们先看看下面的东西
函数指针和指针函数的定义我们看个代码
int *func(int a,int b)
我们之前说过运算符的优先级,「 * 」 的优先级低于「( 」
由于「 * 」的优先级低于「 () 」的优先级,因而func首先和后面的「 () 」结合,也就意味着,func是一个函数。即:
int *(func)(int a,int b)
然后这个函数的返回值的类型是「 int * 」 即:指向int类型的指针
然后我们这样修改上面的代码
int (*func)(int a,int b)
「(*func) 」说明func是一个指针,然后后面跟着「()」说明这个指针指向一个函数,即指向函数的指针。
所以
函数指针: 首先是一个指针,这个指针指向一个函数
指针函数:首先是一个函数,这个函数的返回值一个指针
用typedef声明一个函数指针我们声明一个函数指针,正常方法是
int (*pfunc)(int a,int b)
当我们命名很多个函数指针的时候,用上面的方法显得非常不方便,所以我们可以这样做
typedef int (*PF) (int a,intb) PF pfunc;
例程:
#include "stdio.h" typedef int(*PF)(int, int); int add(int a, int b) { return a + b; } int reduce(int a, int b) { return a - b; } int main() { PF pfunc = NULL; pfunc = add; printf("add:%d\n",pfunc(3, 4)); pfunc = reduce; printf("reduce:%d\n", pfunc(3, 4)); /*getchar是用VS编写方便查看输出*/ getchar(); return 0; }
void (*f[])(char *)
f是个什么鬼东西了?
[]的优先级 比*的优先级高,所以f首先是修饰了数组,然后跟后面的*组合,就说明这个数组里面住的都是 指针,这些指针是什么呢,再出来看看就看到了,这个指针是 一个函数,这个函数的 参数是char *返回值是void。
示例代码
#include void (*f[3])(char *); void efunction(char * s) { printf("%s\n",s); } int main() { f[0] = efunction; //void (*f[])(char *) = {efunction}; (*f[0])("hello code"); return 0; }
代码输出
hello code -------------------------------- Process exited after 0.08441 seconds with return value 0 请按任意键继续. . .函数指针在项目总的实际应用
这是我的android项目hal部分的代码,这部分代码用到的是函数指针,通过name来调用不同的函数。
我们有时候看别人的代码时候,经常是一面懵逼,比如下面这个void (*p)();还有这个(*(void(*) ())0)();
我记得我在以前的文章里面有谈到一个右左原则,从p开始看,往右走直到遇到)再往左走遇到(,(*p)我们就可以看出p是一个指针,继续分析往右走,遇到(),说明p指向一个(void)的函数,往左走,知道p指向的函数返回值是void。
ok,看下面的例子。
#include "stdio.h" void Function() { printf("Call Function!\n"); } int main() { void(*p)(); *(int*)&p = (int)Function; (*p)(); getchar(); return 0; }
然后继续分析(*(void(*) ())0)();
-
1、void (*)() 我们上面分析了这个是一个函数指针,只是把p去掉了而已。
-
2、把上面的void (*)()用PN代替,上面的表达式变成(*(PN)0)();PN后面有一个0,这个是让我们咋舌的地方,然后我们向一下(char)a;这样的表达式,所以*(PN)0就是把0当成一个地址,强制转换为PN类型,用*这个钥匙取出这个地址区域的值。
-
3、把(*(PN)0)()替换成PM,原来的表达式变成PM(),这样大家看起来比较容易了吧,就是正常的函数调用。
#include #include char * fun1(char * p) { printf("%s\n", p); return p; } char * fun2(char * p) { printf("%s\n", p); return p; } char * fun3(char * p) { printf("%s\n", p); return p; } int main() { char * (*pf[3])(char * p); pf[0] = fun1; // 可以直接用函数名 pf[1] = &fun2; // 可以用函数名加上取地址符 pf[2] = &fun3; pf[0]("fun1"); pf[0]("fun2"); pf[0]("fun3"); getchar(); return 0; }
推荐阅读:
专辑|Linux文章汇总
专辑|程序人生
专辑|C语言
嵌入式Linux
微信扫描二维码,关注我的公众号