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

C/C++
176
0
0
2024-02-25

⛺️生活的理想,就是为了理想的生活!

文章目录
  • 📋 前言
  • 💬 函数指针
  • 💭 函数名 和 &函数
  • 💭 函数指针的定义
  • 💻 函数指针pf 和 函数名的关系
  • 💻 函数指针的应用
  • 📜 代码一讲解
  • 📜 代码二讲解
  • 📝全篇总结

📋 前言

🌈hello! 各位宝子们大家好啊,数组指针的概念我们学完了那么今天就给大家打来函数指针的讲解! ⛳️函数既然也有自己的地址那么,是不是也可以用指针来存储呢?答案是是的!今天就来看一下函数指针是什么,以及如何应用!

💬 函数指针

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

💭 函数名 和 &函数

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

📚 代码演示:

#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. 然后进行函数调用 (*)()

在这里插入图片描述

其实这里最难理解的话就是 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;
}

📝全篇总结

✅ 归纳: 好了以上就是关于函数指针是什么,如何定义和应用就全部讲解完毕啦! 函数指针是干嘛的?函数指针的定义函数指针的应用函数指针实例 ☁️ 把本章的内容全部掌握,铁汁们就可以熟练应用函数指针啦!快去练起来吧!

在这里插入图片描述