那些 C语言指针 你不知道的小秘密 (完结篇)

本篇会加入个人的所谓‘鱼式疯言’
❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言
而是理解过并总结出来通俗易懂的大白话,
我会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的.
🤭🤭🤭可能说的不是那么严谨.但小编初心是能让更多人能接受我们这个概念 !!!
在这里插入图片描述

前言

在上一篇指针小秘密的文章中,我们主要理解了一下几点

  • 字符指针 :分别带友友们了解了 单字符 和 字符串 的所对应的字符指针以及使用细节
  • 数组指针 : 介绍了数组指针的概念 和 以及 数组指针 常用场合
  • 二维数组传参的本质 : 理解了二维数组本质的 内核

今天小编将带着大家给我们C语言最重要的指针内容画上一个完美的句号,这次主要的指针主题是咱们的函数指针,下面让我们最后享受我们魔术般的指针秘密吧 💕 💕 💕

目录

  1. 函数指针
  2. 函数指针数组
  3. 函数指针数组的实际运用:转移表
  4. 函数指针的实际运用:回调函数

一. 函数指针

这时就有小伙伴问了,什么 😲 😲 😲
函数居然还有指针 ! ! !

是的,我们函数也有自己专门的地址,而且可以通过指针变量来存储的自身函数的地址的

<1>. 函数指针的简介

C语言中的函数指针是指向函数的指针 变量 。它可以用来存储函数的地址,以便在程序中 调用 该函数。

函数指针的声明格式如下:

  return_type (*pointer_name)(parameter_list);
  • return_type是函数的返回类型
  • pointer_name是函数指针的名称
  • parameter_list是函数的参数列表。

<2>. 举个栗子

小编在上面提过一个东西
那就是我们的函数是否有自己的地址呢,下面让我们来做个测试吧 💖 💖 💖

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

在这里插入图片描述
从中我们可以发现,函数是有地址的,而且聪明的小爱同学已经偷偷告诉我了

他发现 函数名&函数名 地址居然是一样,并提出了函数名就是函数地址的大胆猜想呢 ! ! !

这个猜想到底对不对呢,我们还有待考证,不妨带着问题我们继续研究吧 😁 😁 😁

<3>. 函数指针的声明

void test()
{printf("hehe\n");
}
void (*pf1)() = &test;
void (*pf2)() = test;
int Add(int x, int y)
{return x + y;
}
int(*pf3)(int, int) = Add;
int(*pf3)(int x, int y) = &Add;//x和y写上或者省略都是可以的

宝子们都看到了我们函数声明是这样子的,可他有哪些结构呢,是不是还是很模糊呢,下面请看

鱼式疯言

有图有真相
在这里插入图片描述

<4>. 函数指针变量的实际运用

通过函数指针调用指针指向的函数。

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

在这里插入图片描述

从 pf 和 (*pf) 得出的结果是一致的,这下和小爱同学彻底验证了,我们的函数名就代表地址

鱼式疯言

  1. 函数名相当于数组名一样也表示 地址
  1. int (*) (int ,int)是 函数指针 类型
  1. int (*p)(int ,int ) 是函数指针 变量

二. 函数指针数组

蛙趣,友友们是不是听到这个名字是不是很惊讶呢,那么高级的名字啊,又是函数,又是指针,还是数组的三者结合版啊 !! !

<1>. 函数指针数组的简介

C语言中的函数指针数组是由一组函数指针组成的数组。可以通过函数指针数组来实现函数的动态调用。

函数指针数组的定义方式如下:
// 声明一个函数指针类型
typedef void (*FuncPtr)();
// 声明一个函数指针数组
FuncPtr funcPtrArray[10];

鱼式疯言

函数指针数组本质上是存放地址的 数组,与 函数指针 类似。

<2>. 举个栗子

int add(int x, int y)
{return x + y;
}int sub(int x, int y)
{return x - y;
}int main()
{//声明一个函数指针数组int (*p[2])(int x, int y) = { add,sub };//用 for 循环逐个调用for (int i = 0; i < 2; i++){printf("%d ", p[i](5,4));}return 0;
}

在这里插入图片描述
我们可以利用 函数指针数组 来分别调用不同的函数

鱼式疯言

  1. 函数指针数组本质上是 数组
  1. 竟然是 数组,类型必须一致,包括我们的函数的 参数 以及 返回类型 也是如此

三. 函数指针数组的实际运用:转移表

友友们是否做过简易的 计算器(加减乘除),我想小爱同学一定会这样写

<1>. 一般计算器

//简易计算器普通版
int add(int x, int y)
{return x + y;
}int sub(int x, int y)
{return x - y;
}int mul(int x, int y)
{return x * y;
}int div(int x, int y)
{return x / y;
}void meau()
{printf("******* 1.加法 2.减法 *******\n");printf("******* 3.乘法 4.除法 *******\n");printf("*******  0 . 退  出   *******\n");
}
int main()
{int output = 0;do{int a = 0, b = 0;meau();printf("请选择你要进行的运算:");scanf("%d", &output);int t = 0;switch (output){case 1:printf("请输入两个操作数:");scanf("%d%d", &a, &b);t=add(a, b);printf("%d\n", t);break;case 2:printf("请输入两个操作数:");scanf("%d%d", &a, &b);t = sub(a, b);printf("%d\n", t);break;case 3:printf("请输入两个操作数:");scanf("%d%d", &a, &b);t = mul(a, b);printf("%d\n", t);break;	case 4:printf("请输入两个操作数:");scanf("%d%d", &a, &b);t = div(a, b);printf("%d\n", t);break;case 0:printf("计算器正在退出中...\n", t);break;default:printf("选择错误,请重新选择!\n");break;}} while (output);return 0;
}

在这里插入图片描述
这样写固然没错,但我们有没有更好的调用方法呢

<2>. 转移表

#include<stdio.h>int add(int x, int y)
{return x + y;
}int sub(int x, int y)
{return x - y;
}int mul(int x, int y)
{return x * y;
}int div(int x, int y)
{return x / y;
}int main()
{int output = 0;int (*p[5])(int x, int y) = { NULL,add,sub,mul,div };do{int a = 0, b = 0;meau();printf("请选择你要进行的运算:");scanf("%d", &output);int t = 0;if (output > 0 && output < 5){printf("请输入两个操作数:");scanf("%d%d", &a, &b);printf("%d\n", p[output](a, b));}} while (output);return 0;
}

我们由数组转移到函数的调用的这个过程就叫转移表

所以,我们完美的利用了函数指针数组然后利用其下标引用,可以随时调用我们哪一组函数

鱼式疯言

当我们需要调用同类函数时, 函数指针数组 是个不错的选择哦

四. 函数指针的实际运用:回调函数

啥是回调函数呢,友友们先带着这个疑问来探究我们的今天要学的 回调函数 哦 ! ! !

<1>. 回调函数的简介

在C语言中,回调函数 是指一个函数作为参数传递给另一个函数,并且在后者执行过程中被调用的函数。

<2>. 举个栗子

就拿我们上面这个栗子说明吧,我们的普通版本的计算器是不是还可以利用 函数指针 来优化

//回调函数#include<stdio.h>//利用函数指针传参
void calc(int (*pcalc)(int x, int y))
{int m = 0, n = 0;scanf("%d%d", &m, &n);int fault = pcalc(m, n);printf("%d\n", fault);return 0;
}int add(int x, int y)
{return x + y;
}int sub(int x, int y)
{return x - y;
}int mul(int x, int y)
{return x * y;
}int div(int x, int y)
{return x / y;
}void meau()
{printf("******* 1.加法 2.减法 *******\n");printf("******* 3.乘法 4.除法 *******\n");printf("*******  0 . 退  出   *******\n");
}int main()
{int intput=0;do{meau();scanf("%d", &intput);switch (intput){case 1:calc(add);break;case 2:calc(sub);break;case 3:calc(mul);break;case 4:calc(div);break;case 0 :printf("正在退出中!\n");break;default:printf("选择失败,请重新选择\n");break;}} while (intput);return 0;
}

在这里插入图片描述
小伙伴有没有发现,我们的传递函数时,是用什么来接收的呢,答案应该很明朗了吧,就是我们本篇文章的主角:函数指针

函数指针的出现让我们能够不断重新调用我们重复类型的函数

鱼式疯言

函数 作为 实际参数 时,我们就可以用 函数指针 来作为 形式参数 来接收

函数指针虽好,可不要贪杯哦,他好像我们函数指针数组一样,是要保持 类型统一性

函数指针的优势:可以不用想普通的计算器一样反复调用同样多行语句,从而减少我们代码过多的 冗长

比如
在这里插入图片描述

在这里插入图片描述

总结

  • 函数指针: 理解函数指针的如何声明,为回调函数做好铺垫
  • 函数指针数组:熟悉了函数指针的具体的场景同时也为转移表做好了铺垫
  • 函数指针数组的实际运用:转移表 ——> 真正去贴合实际去理解我们函数指针数组一般的运用场景
  • 函数指针的实际运用:回调函数 ——> 让我们真正理解了当函数作为参数的实际运用

如果觉得小编写的还不错的咱可支持 三连 下 (定有回访哦) , 不妥当的咱请评论区 指正

希望我的文章能给各位宝子们带来哪怕一点点的收获就是 小编创作 的最大 动力 💖 💖 💖

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/467526.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

441. Arranging Coins( 排列硬币)

问题描述 你总共有 n 枚硬币&#xff0c;并计划将它们按阶梯状排列。对于一个由 k 行组成的阶梯&#xff0c;其第 i 行必须正好有 i 枚硬币。阶梯的最后一行 可能 是不完整的。 给你一个数字 n &#xff0c;计算并返回可形成 完整阶梯行 的总行数。 问题分析 等差数列求和问…

任务调度

1.学习目标 1.1 定时任务概述 1.2 jdk实现任务调度 1.3 SpringTask实现任务调度 1.4 Spring-Task 分析 1.5 Cron表达式 https://cron.qqe2.com/ 2. Quartz 基本应用 2.1 Quartz 基本介绍 2.2 Quartz API介绍 2.3 入门案例 <dependency> <groupId>org.springframe…

MongoDB技术架构详解

MongoDB是一个非关系型数据库&#xff0c;以其高性能、可扩展性和灵活性而闻名。MongoDB的技术架构为其提供了强大的数据存储和处理能力&#xff0c;支持各种现代应用程序的需求。本文将深入探讨MongoDB的技术架构&#xff0c;帮助您更好地理解其内部工作原理。 一、MongoDB概述…

【JavaEE】_网络层

目录 1. 网络层的工作 2. IP数据报 3. 地址管理 3.1 IP地址不够的问题 3.1.1 方案一&#xff1a;动态分配 3.1.2 方案2&#xff1a;NAT机制&#xff08;网络地址转换&#xff09; 3.1.2.1 将IP地址划分为2类 3.1.2.2 NAT转换的实现 3.1.3 方案3&#xff1a;IPV6地址 …

如何写出别人写不出的内容(译)

&#xff08;译者序&#xff1a;这篇文章不只是写作&#xff0c;对信息获取、阅读也都有启发。随着社交媒体和 AI 的发展&#xff0c;人们越来越被动的接收海量信息&#xff0c;如何主动查找与整理对自己有用的内容&#xff0c;将是一个不可或缺的能力。&#xff09; 原文&…

VMware清理拖拽缓存 Ubuntu硬盘情况占用分析

这两天在尝试编译Linux源码&#xff0c;我在win上将源码下载下来然后复制到ubuntu上&#xff0c;这一步我粗略看到了三种方法&#xff1a;安装VM tools&#xff0c;就可以使文件正常的在win和ubuntu中复制剪切&#xff1b;使用scp命令将win和linux系统链接起来&#xff1b;使用…

一文彻底搞懂布隆过滤器

文章目录 1. 基本原理2. 布隆过滤器的优点3. 布隆过滤器的缺点4. 布隆过滤器的应用场景 布隆过滤器&#xff08;Bloom Filter&#xff09;是一种空间高效的概率数据结构&#xff0c;用于判断一个元素是否在一个集合中。它使用位数组和一系列哈希函数来实现。 1. 基本原理 首先…

数学之函数的基础性内容的学习

函数是一个很重要的内容 无数的科学家为其进行前赴后继 伽利略&#xff08;比萨斜塔“高空抛物”&#xff09;&#xff0c;笛卡尔&#xff0c;牛顿&#xff0c;莱布尼兹&#xff0c;约翰伯努利&#xff0c;欧拉&#xff0c;傅里叶&#xff0c;迪利克雷&#xff08;德国数学家…

《Linux 简易速速上手小册》第3章: 文件系统与权限(2024 最新版)

文章目录 3.1 Linux 文件系统结构3.1.1 重点基础知识3.1.2 重点案例&#xff1a;设置一个 Web 服务器3.1.3 拓展案例 1&#xff1a;日志文件分析3.1.3 拓展案例 2&#xff1a;备份用户数据 3.2 理解文件权限3.2.1 重点基础知识3.2.2 重点案例&#xff1a;共享项目文件夹3.2.3 拓…

Java之拦截器interceptor

1. 概念 2. 步骤 第一步 第二步 参考资料 https://www.bilibili.com/video/BV1m84y1w7Tb?p168&vd_source705343a89f38d5c0d895383ccf38a5d6

单调队列优化DP问题

目录 1.滑动窗口 2.最大子序和 3.旅行问题 4.烽火传递 5.绿色通道 6.修剪草坪 7.理想的正方形 1.滑动窗口 154.给定一个大小为 n≤106 的数组。 有一个大小为 k 的滑动窗口&#xff0c;它从数组的最左边移动到最右边。 你只能在窗口中看到 k 个数字。 每次滑动窗口向…

理解JAVA EE设计模式

理解JAVA EE设计模式 在Web应用程序的设计和开发阶段,开发人员在开发类似的项目时可能会遇到相似的问题。每名开发人员可能会遇到的问题找出不同或相似的解决方案。但是,这导致一些时间和精力浪费在为相似的问题寻找解决方案上。因此,要啊节省时间和精力,需要记录常见问题…