函数指针和函数

函数指针和函数

信号处理函数简析

void (*signal(int signum, void (*handler)(int)))(int)

如何分析该xxx,甚至不知道是函数还是指针,还是什么其他的变量!
1) 确定是什么类型:

    void (*)(int)

为一个返回值为void,参数为int类型的函数指针!
2) 独立出复合类型:

类型           函数名(参数1, 参数2)
void (*)(int) signal(int signum, void (*handler)(int))

3) 总结:

    为一个名为signal的函数,参数为int signum & 函数指针(信号处理函数) void (*handler)(int);返回值为void (*)(int)

疑问:
这里通过第一步确定了其中为一个函数指针类型,但是最终分析出来的是一个函数类型?函数和函数指针类型有什么区别?

函数指针和函数

函数名

void myfun(int x);      //函数
void* myFun(int x);     //指针函数,返回值是指针的函数
void (*myFun)(int x);   //函数指针,返回值是void,参数是int的函数指针,**是数据类型**;可以写成:
                        // void (*)(int) myFun;
Eg1:
#include <stdio.h>

void my_fun(int);
void (*my_fun_p)(int);

void my_fun(int x) {
    printf("%d\n", x);
}

int main(int argc, char *argv[]) {
    my_fun(10);
    my_fun_p = &my_fun;     //将函数名地址赋值为函数指针
    (*my_fun_p)(20);
    return 0;
}
Eg2:
#include <stdio.h>

void my_fun(int);
void (*my_fun_p)(int);

void my_fun(int x) {
    printf("%d\n", x);
}

int main(int argc, char *argv[]) {
    my_fun(10);
    my_fun_p = my_fun;      //将函数名直接赋值给函数指针
    (*my_fun_p)(20);
    return 0;
}
反汇编:
Dump of assembler code for function main:
   0x000000000040054e <+0>: push   %rbp
   0x000000000040054f <+1>: mov    %rsp,%rbp
   0x0000000000400552 <+4>: sub    $0x10,%rsp
   0x0000000000400556 <+8>: mov    %edi,-0x4(%rbp)
   0x0000000000400559 <+11>:    mov    %rsi,-0x10(%rbp)
=> 0x000000000040055d <+15>:    mov    $0xa,%edi
   0x0000000000400562 <+20>:    callq  0x40052d <my_fun>
   0x0000000000400567 <+25>:    movq   $0x40052d,0x200ad6(%rip)        # 0x601048 <my_fun_p>
   0x0000000000400572 <+36>:    mov    0x200acf(%rip),%rax        # 0x601048 <my_fun_p>
   0x0000000000400579 <+43>:    mov    $0x14,%edi
   0x000000000040057e <+48>:    callq  *%rax
   0x0000000000400580 <+50>:    mov    $0x0,%eax
   0x0000000000400585 <+55>:    leaveq 
   0x0000000000400586 <+56>:    retq
发现Eg1 & Eg2最终反汇编后的代码完全一致,看来函数名和函数指针有一些共同的联系

1)其实,my_fun的函数名与my_fun_p函数指针都是一样的,即都是函数指针。my_fun函数名是一个函数指针常量,而my_fun_p是一个函数数指针变量,这是它们的关系。
2)但函数名调用如果都得如(*my_fun)(10)这样,那书写与读起来都是不方便和不习惯的。所以C语言的设计者们才会设计成又可允许my_fun(10)这种形式地调用(这样方便多了并与数学中的函数形式一样,不是吗?)。
3)为统一起见,my_fun_p函数指针变量也可以my_fun(10)的形式来调用。
4)赋值时,即可my_fun_p = &my_fun形式,也可my_fun_p = my_fun。

函数指针的使用

1)函数分析:
再次分析以上的信号处理函数:

void (*signal(int signum, void (*handler)(int)))(int)

所以不要纠结于是函数还是函数指针了,只不过一个是常量指针,一个是变量指针;

展开后:

void(*)(int) signal(int signum, (void(*)(int)) handler);

最终都是标示的为函数对应堆栈的入口地址,如果是函数,后面会有相关的参数列表,会在最终编译的时候作为参数压入堆栈中,即使是void类型也会;如果没有(),即函数指针类型:
void (*)(int) signal;
则就是定义了一个函数指针类型,如果赋值为函数入口地址后,也可以和函数名一样使用。

2)实际的函数指针作为参数的实例:回调函数的实现
给你一个实例:
要求:我要设计一个CallMyFun函数,这个函数可以通过参数中的函数指针值不同来分别调用MyFun1、MyFun2、MyFun3这三个函数(注:这三个函数的定义格式应相同)。
实现:代码如下:

/* 自行包含头文件 */
void MyFun1(int x);
void MyFun2(int x);
void MyFun3(int x);
typedef void (*FunType)(int ); /* ②. 定义一个函数指针类型FunType,与①函数类型一致 */
void CallMyFun(FunType fp,int x);
int main(int argc, char* argv[])
{
   CallMyFun(MyFun1,10); /* ⑤. 通过CallMyFun函数分别调用三个不同的函数 */
   CallMyFun(MyFun2,20);
   CallMyFun(MyFun3,30);
}
void CallMyFun(FunType fp,int x) /* ③. 参数fp的类型是FunType。*/
{
   fp(x);/* ④. 通过fp的指针执行传递进来的函数,注意fp所指的函数是有一个参数的。 */
}
void MyFun1(int x) /* ①. 这是个有一个参数的函数,以下两个函数也相同。 */
{
   printf("函数MyFun1中输出:%d\n",x);
}
void MyFun2(int x)
{
   printf("函数MyFun2中输出:%d\n",x);
}
void MyFun3(int x)
{
   printf("函数MyFun3中输出:%d\n",x);
}

输出结果:略分析:看我写的注释。你可按我注释的①②③④⑤顺序自行分析。

参考博客:http://c.biancheng.net/cpp/html/496.html

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值