【C语言进阶篇】数组指针都学过了那么函数指针你了解嘛?


在这里插入图片描述

鸽芷咕:个人主页

  个人专栏:《C语言初阶篇》 《C语言进阶篇》

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

文章目录

  • 前言
  • 函数指针
    • 函数名 和 &函数
    • 函数指针的定义
      • 函数指针pf 和 函数名的关系
  • 函数指针的应用
      • 代码一讲解
      • 代码二讲解
  • 全篇总结

前言

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

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

函数指针

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

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

函数名 和 &函数

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

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

代码演示:

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

代码结果:
【C语言进阶篇】数组指针都学过了那么函数指针你了解嘛?_第1张图片

函数指针的定义

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

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

✅总结:

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

代码演示:

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

代码结果:
【C语言进阶篇】数组指针都学过了那么函数指针你了解嘛?_第2张图片

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

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

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

代码演示:

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

代码结果:
【C语言进阶篇】数组指针都学过了那么函数指针你了解嘛?_第3张图片

函数指针的应用

阅读两段有趣的代码:

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

代码一讲解

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

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

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

【C语言进阶篇】数组指针都学过了那么函数指针你了解嘛?_第4张图片

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

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

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

全篇总结

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

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

你可能感兴趣的:(《C语言进阶篇》,c语言,开发语言,C++,代码实例)