C语言、嵌入式重点知识:回调函数

前言

上文分享了一个专用的双链表的基本操作示例:双链表的操作示例(附代码)

这里提到了一个关键词:专用。与专用对应的词是通用

我们从字面上可以很容易理解这两个词,专用就是针对特定情况的,特点就是很有局限性。

通用就是可以针对大多数情况(更理想的就是所有情况),特点就是适用性广。

为什么说上篇笔记的双链表是专用的?

从我们的定义的元素数据类型就可以知道,我们这个双链表是只是用来存储int类型的数据的,这就很能体现出了局限性(这只是其中一点,当然还有其它的很多局限性),因此是个专用的双链表。

我们要编写一个通用的双链表的话,我们首先要做的是就是修改双链表结点结构体了,可以修改为:

如果我们要存放整数,我们可以把void*强制转换成整数使用。当然这篇笔记的重点不是分享通用的双链表。

我们这篇笔记要分享的是回调函数,下面进入重点内容:

回调函数法 VS 常规法

我们上篇笔记中有一个打印输出链表数据的函数:

这是我们这个专用的双链表中打印链表数据函数,我们存储的是整数,所以用%d打印。那么,如果我们面向的是通用的双链表呢?

我们无法预知其中的数据,可能是整数,也可能是字符串,或者是其它的数据。那么怎么办呢?这里有几种方法:

方法一:实现多个函数,需要用到哪个就调哪个

比如存放的是整数,可以调用dlist_print_int函数来打印;存放的是字符串,可以调用dlist_print_string函数来打印。

这种方法很简单,但有个缺点:每个函数都很相似,会有大量重复的代码。

方法二:传入一个附加的参数来选择打印的方式

这种方法使用一个参数来选择打印的方式。避免了方法一中产生大量重复的代码的问题。

但是我们每当要增加新类型时,都得修改这个dlist_print函数,对于一个通用的双链表来说,这样的修改是不够好的。

这里dlist_print函数也是通用双链表的一部分,我们应该尽量少去修改它。

假如我们把一个通用的双链表的基础操作比喻做一栋楼房的地基,地基一旦牢牢固固的搭好之后,我们就不要再去动它了,应该把精力放在如何搭建房子的上层上。

方法三:回调函数法

上面两种方法应该是很容易想到的方法。现在来分享我们可能想不到的方法——回调函数法,这也是本篇笔记要分享的重点。

可能有很多朋友没用过回调函数,甚至有些朋友都没听说过。这里先简单介绍回调函数的一些概念(以下概念来自百度百科):

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。

回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

知识点:变量指针指向的是一块数据,指针指向不同的变量,则取到的是不同的数据;函数指针指向的是一段代码(即函数),指针指向不同的函数,则具有不同的行为。

回归正题,下面看如何使用回调函数法来实现上面的需求。

首先,我们需要实现一个通用的打印函数dlist_print,把函数指针变量作为其中一个参数传入。

其次,我们调用者得根据实际情况实现一个用于打印的回调函数,这里我们实现的的回调函数是dlist_print_int

最后,在用到打印的地方调用dlist_print函数即可。

用回调函数法是不是很巧妙?

此处,我们用到了typedef来“封装”一个打印链表数据的函数指针类型,这可能会刷新了初学者对于typedef关键字的认识。

因为我们刚开始学C语言的时候,总认为typedef取别名的一般形式为:

typedef  旧名字  新名字;

确实也是这样,但遇到给函数指针类型、数组类型等定义别名的时候就要特别区分了。如:

typedef char ARRAY20[20];
ARRAY20 a1,a2; /* 等价于char a1[20],a2[20]; */

别问为什么,就是这样的。。。

回调函数的例子

上面分析了那么多,可能很多朋友会觉得回调函数太麻烦了,没必要用。但是现实是,回调函数在我们的C编程、嵌入式编程中用得很广泛。

1、在C编程中

在C语言的通用工具库stdlib.h中,有如下一个函数原型:

void qsort(void *, size_t, size_t, int (comp*)(const void *, const void *));

这是在C通用工具库中声明的一个快速排序算法函数,其可以用来排序int类型、float类型以及字符串数据。

可以按从小到大的顺序也可以按从大到小的顺序排序。其关键在于函数指针comp指向的函数的具体实现。

2、在嵌入式编程中

我们之前的笔记:【RT-Thread笔记】PIN设备中断配置中,就有用到回调函数。

RT-Thread给我们提供了PIN设备中断回调绑定函数:rt_pin_attach_irq

这是个中断实验,产生中断会回调我们的回调函数,所以可以在在我们的回调函数里做一些产生中断后需要做的操作。

比如我们在这个中断回调里打印一串字符串。每当中断来时,就会打印该字符串:

总结

回调函数是一个很重要的知识点,我们需要掌握。而回调函数又与函数指针联系密切,我们要努力把函数指针弄懂、用熟。

在C语言中,指针很重要,函数指针更重要。正如前辈们常说类似这样子的话:不会C指针,就没学会C语言;不会函数指针,就不要称自己是C语言高手。

在这几种方法中的分析中,其实回调函数更多的是体现出了软件分层的思想。分层思想在我们软件开发中是一种很重要的思想,简单的分层我们都会,但是怎么才能算是分层分得很好呢?

那就是不该动的地方不动,该动的地方才动,衔接得很好,就像上面的回调函数法。

对于编程的学习,关于编程语言的学习,知识点就是那么多,很快就能学完了,但是真正灵活的、熟练应用起来真的是不容易,这需要我们大量地分析、思考、练习。

有时间的话我们也应该多读读一些关于软件设计思想的书籍,这也是我最近在读的一类书,学学前辈们总结出的一些精华知识。

最后

以上就是本次关于回调函数的笔记分享,如有错误,欢迎指出。如果觉得文章不错,转发、在看,也是我们继续更新得动力。


  回复「 篮球的大肚子」进入技术群聊

回复「1024」获取1000G学习资料

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

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

相关文章

python神经网络预测结果每次不一样_神经网络预测

神经网络预测时间:2019-12-09 12:34:00 作者:路由君 来源:路由器之家路由器之家网今天精心准备的是《神经网络预测》,下面是详解!bp神经网络预测是不是数据越多,预测能力就越好?不仅是神经网络,…

js数组的拷贝赋值复制二三事总结

今天在看React-native性能优化的时候,看到如何避免shouldComponentUpdate的异常数据时,脑内一阵风暴,从而牵连出一连串的问题,于是有了这一篇关于js数组的复制(深浅拷贝)与赋值等为何能产生异常数据的文章。…

今天我勇敢的点就一个gpio口

现在已经三月份了,时间过得超快,早上起来打开电脑,有点不习惯,微信群唧唧歪歪的那些股神今天不知为什么安静了。我喜欢看大家热闹的样子,更喜欢热闹的时候给我们发几个红包。我记得2015年,股市非常好&#…

Linux 通用gpio口驱动,rockchip

dts文件 gpio_rs485: gpio_rs485 {status = "okay";compatible = "gpio,px30-gpio";cname = "rs485";en-gpio = <&gpio3 12 GPIO_ACTIVE_HIGH>;}

android自定义控件

---恢复内容开始--- 1.新建一个类&#xff0c;继承View父类。重写一个或多个构造器后&#xff0c;在图编辑器里就有该控件可以拖动添加了。 2.若想使用Draw来绘制自己的控件。可以在View&#xff08;&#xff09;方法中使用Draw&#xff08;&#xff09;来绘制。 3.还可以定义控…

python语言中strike_Python学习笔记

嵌套函数作用域def make_adder(augend):def add(addend):return augend addendreturn add内部的函数可以访问外部函数scope内的变量&#xff0c;但是不能够重新对其赋值。如果重新赋值那么会在内部函数的scope内创建一个同名的本地变量(Python不允许对non-local变量赋值)。yie…

数据结构(4)

文章目录栈与队列栈队列![在这里插入图片描述](https://img-blog.csdnimg.cn/20200301182116946.png?x-oss-processimage/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl80MzY2Mzc4MA,size_16,color_FFFFFF,t_70)栈与队列 栈 ha…

写一个公用的gpio口驱动

因为项目需要控制的GPIO口比较多&#xff0c;如果每个GPIO口都写一个驱动就显得比骄麻烦&#xff0c;所以就写了一个通用的GPIO口驱动。只要dts里面配置好设备GPIO相关信息就可以自动加载了。可以很充分的体现多个设备一个驱动的优良性。dts文件gpio_rs485: gpio_rs485 {status…

数据结构(5)

文章目录各种算法选择排序插入排序希尔排序***快速排序***归并排序二分查找各种算法 def bubble_sort(alist):"""冒泡排序"""n len(alist)for j in range(n-1):count 0for i in range(0, n-1-j):# 班长从头走到尾if alist[i] > alist[i1]:a…

大神们都应该去哪里工作?

接上一篇文章。文中提到我认识了一个做嵌入式的大神&#xff0c;技术真的非常厉害。我们在调试过程中&#xff0c;遇到问题&#xff0c;他总是能告诉我们排查的手段&#xff0c;而且针对一个问题&#xff0c;他能想到几种不同的解决方案。嵌入式驱动跟其他软件有点不一样&#…

堆喷射

堆喷射主要用于绕过ASLR。下面演示堆喷射分析与效果。 1.代码 void heap_spray(){   char chunk[LEN] { 0 };   memset(chunk, 0x90, LEN - 10);   strcat(chunk, "shellcode");   for (int i 0;i < 100;i)   {     void *p malloc(LEN);    …

大华管理平台用户名_智能财务引领商业与管理变革——浙大EMBA首席财务官研习社走进大华股份...

杭商传媒记者 周 珂/文 徐青青/摄编辑 何影丹当前经济环境和市场需求瞬息万变&#xff0c;企业竞争日益激烈&#xff0c;盈利持续下降&#xff0c;产品、服务同质化严重&#xff0c;这不仅对业务部门提出了挑战&#xff0c;也对传统的财务管理产生了巨大的冲击。财务不能高…

【深度】韦东山:一文看尽 linux对中断处理的前世今生

作者&#xff1a;韦东山前言&#xff1a;本文&#xff0c;4200字&#xff0c;研究代码花了一天&#xff0c;写出来花了一天&#xff1b;录视频估计又得花半天&#xff1b;真怀念以前简单粗暴的生活啊&#xff1a;拿起话筒就录视频&#xff0c;先画好图&#xff1f;那是不需要的…

数据结构(6)二叉树

文章目录二叉树二叉树三种遍历二叉树知中及先或后序&#xff0c;求二叉树二叉树 class Node(object):""""""def __init__(self, item):self.elem itemself.lchild Noneself.rchild Noneclass Tree(object):"""二叉树"&qu…

鼠标中间无法打开新标签_还记得鼠标有几个键?Win10环境鼠标中键的妙用

大家经常使用鼠标&#xff0c;但有多少人还能意识到&#xff0c;其实鼠标是有三个键的&#xff1f;除了常用的左键右键&#xff0c;鼠标中间的滚轮也可以按下去&#xff0c;然而鼠标的这个中键&#xff0c;却日常坐冷板凳&#xff0c;几乎被人所遗忘。其实&#xff0c;鼠标中键…

vue技术栈

1 vue 说明:vue生命周期&#xff1a;技术点&#xff1a;1&#xff1a;常用的API:computed&#xff0c;methods&#xff0c;props&#xff0c;mounted&#xff0c;created&#xff0c;components 2vue-cli说明:vue绞手架&#xff0c;用于快速搭建项目&#xff0c;并管理项目技术…

python中pillow库怎么使用_Python 图像库Pillow使用

需求是将一张含有透明度的图片和一张正常图片进行常规合并&#xff0c;并在图片上添加文字&#xff0c;代码如下&#xff1a;from PIL import Imageimport cv2base_image Image.open(图片的绝对路径) # 有透明度的图片target Image.new(RGB, base_image.size, (0, 0, 0, 0)) …

找不到问题的核心,你永远解决不了问题。

上学那会儿&#xff0c;我是个数学常拿60分的人&#xff08;满分150分&#xff09;&#xff0c;我却喜欢做几何题。根据已知的条件&#xff0c;推导出所需的其他条件&#xff0c;所有需要的条件得出后&#xff0c;再根据正确的公式算出所求的值。这个不断求解的过程其实挺有意思…

.net MVC路由

首先&#xff0c;在MVC应用程序的入口函数Application_Start()中(在事件在GLOBAL文件中)&#xff0c;通过RouteConfig.RegisterRoutes(RouteTable.Routes)注册路由集合信息。RouteTable.Routes是一个公开的路由集对象&#xff0c;用于保存路由信息集合信息&#xff0c;类型问Ro…

浏览器访问pdf 的url怎么加密_怎么解决加密的pdf文件?

加密版的PDF文件我们每次打开时都需要输入密码才能进行阅读&#xff0c;而且一般不能直接进行复制和打印&#xff0c;这样会非常的麻烦。那如果我门想要解决这一问题呢就要将pdf文件的密码进行解除&#xff0c;可是如果是平民的我们不会解决怎么办呢&#xff1f;毕竟我们又不是…