【C语言进阶篇】函数指针的全面详解

举报
鸽芷咕 发表于 2023/09/19 21:00:56 2023/09/19
【摘要】 函数指针的概念你了解嘛?他是怎么调用的以及实例应用是怎么样的?本篇文章给你全部讲解清楚,让你轻松学会函数指针!

@TOC

📋 前言

  🌈hello! 各位宝子们大家好啊,数组指针的概念我们学完了那么今天就给大家打来函数指针的讲解!
  ⛳️函数既然也有自己的地址那么,是不是也可以用指针来存储呢?答案是是的!今天就来看一下函数指针是什么,以及如何应用!
  📚本期文章收录在《C语言进阶篇》,大家有兴趣可以看看呐
  :tent: 欢迎铁汁们 :heavy_check_mark: 点赞 👍 收藏 ⭐留言 📝!

🔥 ==注:vs2022 等C语言学习工具都在《学习工具专栏》。 还有各种实用调试技巧有兴趣可以去看看!==

💬 函数指针

函数指针是什么其实很简单?我们依旧可以类比一下

  • 数组指针 —— 指向数组的指针
  • 函数指针 —— 指向函数的指针

💭 函数名 和 &函数

函数既然有地址那么该怎么拿到函数的地址,数组名是数组的地址,&数组可以拿到整个数组的地址!那么函数名是函数的地址嘛?

  • 其实函数名是函数的地址
  • &函数名也是函数的地址

📚 代码演示:

#include <stdio.h>
void test()
{
    printf("hehe");
}
int main()
{
    printf("%p\n", &test);
    printf("%p\n", test);
    
    return 0;
}

📑 代码结果:
在这里插入图片描述

💭 函数指针的定义

数组指针我们知道他的指针类型是 int (*)[ ] 这样接收的,那么函数指针该怎么样书写呢?假设我们的函数指针类型的

  • 那么就应该这样书写
  • int (*pf)(int, int ) = test;
  • * ==表明了他是个指针,是用 pf 接收的函数地址==
  • int ()(int , int ) ==说明了他是一个函数==
  • ==第一个int是他的返回值,括号里面的是参数类型==
  • 这和数组指针是不是非常类似int (*pf)[ ]= &arr;
  • * 表明了他是个指针,是用 pf 接收的数组地址
  • int [ ] 说明了他是一个数组

✅总结:

  • int (* )(int, int)是函数指针类型
  • int (*)[ ] 是数组指针类型

📚 代码演示:

#include <stdio.h>
int add(int x, int y)
{
    return x + y;
}
int main()
{
    int (*pf)(int, int) = add;
    printf("%d\n", add(2, 3));
    printf("%d\n", (*pf)(2, 3));
    return 0;
}

📑 代码结果:
在这里插入图片描述

💻 函数指针pf 和 函数名的关系

大家看啊!既然我们是用 pf 接收的函数地址,而函数名本身代表的是函数地址。所以我们可不可以说 pf 等于函数名!

  • int (*pf)(int, int) = add;
  • 所以 pf == add
  • 那么我们在用指针解引用调用函数时,是不是可以直接写
  • pf(2,3) == add(2,3)我们来试一试

📚 代码演示:

#include <stdio.h>
int add(int x, int y)
{
    return x + y;
}
int main()
{
    int (*pf)(int, int) = add;
    printf("%d\n", add(2, 3));
    printf("%d\n", pf(2, 3));
    return 0;
}

📑 代码结果:
在这里插入图片描述

💻 函数指针的应用

阅读两段有趣的代码:

  • 这俩句代码是干嘛呢?
//代码1
(*(void (*)())0)();
//代码2
void (*signal(int , void(*)(int)))(int);

📜 代码一讲解

其实代码一 (*(void (*)())0)(); 是调用0地址处的函数,

  1. 将 0 转换为函数指针类型 void (*)()
  2. 然后进行函数调用 (*)()

🔥 ==注:此代码来自于《C的陷阱与缺陷》==

在这里插入图片描述

其实这里最难理解的话就是 0 这个数字,这段代码我们只能先从 0 这个数字下手。

  • 大家看 0x00ff40 我们可以理解为**16**进制的整数!
  • 当然 我们也可以理解 0x00ff40 是个地址毕竟地址也是编号嘛!是以 16 进制编写的。

这时数字 0 就好理解了,我们可以把 0 当成一个地址里面,默认他是个整形。所以那么把地址 0 强制转换为一个函数指针

  • 然后再进行解引用调用这个函数
  • (*(void (\*)())0)();

📜 代码二讲解

其实代码二void (*signal(int , void(*)(int)))(int); 可以这样理解,

  1. signal 是 一个函数声明signal(int , void(*)(int))
  2. signal 函数有 俩个参数,第一个参数的类型是int ,第二个参数的类型是 void(*)(int) 函数指针类型
  3. 该函数指针指向的函数 有一个 int 类型的参数,返回类型是 void
  4. signal 函数的返回类型也是一个 函数指针类型void (*signal(int , void(*)(int)))(int);
  5. 该函数指针 指向的函数 有一个int 类型的参数,返回类型是 void

这样看起来这段代码是不是不好理解,那么我们对指针类型重名一下就看的简洁好懂了!

#include <stdio.h>

int main()
{
    typedef void(*pf_t)(int);
    pf_t signal(int, pf_t);

    void (*signal(int, void(*)(int)))(int);
 
    return 0;
}

📝全篇总结

✅ 归纳:
好了以上就是关于函数指针是什么,如何定义和应用就全部讲解完毕啦!
  函数指针是干嘛的?
  函数指针的定义
  函数指针的应用
  函数指针实例
:cloud: 把本章的内容全部掌握,铁汁们就可以熟练应用函数指针啦!快去练起来吧!
看到这里了还不给博主扣个:
⛳️ 点赞:sunny:收藏 :star: 关注

💛 💙 💜 ❤️ 💚💓 💗 💕 💞 💘 💖
拜托拜托这个真的很重要!
你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。
在这里插入图片描述

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。