你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

一个有意思的题目——函数指针,指针函数,函数指针数组

[复制链接]
gaosmile 发布时间:2020-9-12 14:59

这是一个群友发的笔试题目,里面涉及的东西也比较有意思。

直接看代码

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;
}
微信图片_20200912145623.png img再说回上面的那个题目void (*f[])(char *)

f 是个什么鬼东西?

[] 的优先级 比 *的优先级高,所以 f首先是修饰了数组,然后跟后面的 *组合,就说明这个数组里面住的都是 指针,这些指针是什么呢,再出来看看就看到了,这个指针是 一个函数,这个函数的 参数是 char *返回值是void。

示例代码

#include <stdio.h>

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来调用不同的函数。

微信图片_20200912145627.png
android hal部分代码懂了这些,我们就可以看懂别人的代码了

我们有时候看别人的代码时候,经常是一面懵逼,比如下面这个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;
}
微信图片_20200912145631.png 输出结果

然后继续分析(*(void(*) ())0)();

  • 1、void (*)()  我们上面分析了这个是一个函数指针,只是把p去掉了而已。
  • 2、把上面的void (*)()用PN代替,上面的表达式变成(*(PN)0)()N后面有一个0,这个是让我们咋舌的地方,然后我们看一下 (char)a;这样的表达式,所以*(PN)0就是把0当成一个地址,强制转换为PN类型,用*这个钥匙取出这个地址区域的值。
  • 3、把(*(PN)0)()替换成PM,原来的表达式变成PM(),这样大家看起来比较容易了吧,就是正常的函数调用。
给个例子自己去参透一下#include <stdio.h>
#include <string.h>

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;
}

收藏 评论0 发布时间:2020-9-12 14:59

举报

0个回答

所属标签

STM32团队

意法半导体微控制器和微处理器拥有广泛的产品线,包含低成本的8位单片机和基于ARM® Cortex®-M0、M0+、M3、M4、M33、M7及A7内核并具备丰富外设选择的32位微控制器及微处理器


最新内容

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版