历史上的今天
返回首页

历史上的今天

今天是:2025年12月21日(星期日)

2022年12月21日 | 单片机怎么用回调函数在不同文件之间传递数据

2022-12-21 来源:zhihu

我们先来理解一下回调函数的作用。

函数我一般喜欢分为输出型和输入型(个人理解)。

输出型:

就是我们主动去调用的控制函数,比如说控制LED灯去亮和灭,控制蜂鸣器响和不响,控制LCD显示,控制继电器吸合和断开。

简单来说,就是我们知道什么时候该去调用这些函数,比如说满足某些条件的时候,我们就会主动去调用这些函数。

这种函数,就是输出型函数。

输入型:

输入型函数一般是用在不同.c文件/不同层(硬件层、应用层)之间传递信号和数据的,比如说按键检测、串口数据。

我们不知道什么时候按键会被按下、什么时候串口会有数据过来对吧?

当然,我们可以写一个带返回值的函数,然后定时去检测,比如说定时10ms去扫描一下按键。

unsigned char ScanKey(){
  //按键检测程序…}

然后我们在主程序用:

while(1){
  unsigned char key;if(10ms时间到){Key = ScanKey();}
       if(Key == 有效按键值)
       {
              //执行按键功能程序}}

这样不断地去扫描按键,检测按键是否被按下。

这种方式当然也是可以的,只是不够专业,不够好。

因为这个我需要一直在while循环里判断Key的值,然后根据Key的值来判断有没有按键按下,在一定程度上,造成了cpu资源的浪费。

而且有些应用场景,这种方式不好实现,比如说串口数据,你不能一直在while循环里判断是否有新的串口数据过来吧?

那我们理想的一种状态是什么?

就是如果有按键按下了,或者有新的数据来了,再通知我。

这种通知方式一般叫事件触发,就是触发了按键这个事件,我才去处理。

所以,这个时候回调函数就能很好地解决这种需求。

我们还是拿按键来举例。

前面我说每个人写回调函数的风格可能都不一样,STM32固件库的那些中断处理函数基本都是回调函数,但是跟我的编写风格还是有些差异。

我们在写回调函数的时候,需要以下几步:

第一步:

自定义一个函数指针类型,类型名称是KeyEvent_CallBack_t。

typedef void (*KeyEvent_CallBack_t)(KEY_VALUE_TYPEDEF keys);

还有这个一般是要自定义在头文件,因为别的.c文件也会用到。

这是一个无返回值的形参是KEY_VALUE_TYPEDEF枚举类型函数指针类型

一般这个形参keys就是我们最终要通过回调函数传递到别的.c文件的信号/数据,如果是按键检测的话也就是按键值,是哪个按键按下的。

我们来看下KEY_VALUE_TYPEDEF这个枚举都有哪些值?

typedef enum{
       KEY_IDLE_VAL,
       KEY1_CLICK,
       KEY1_CLICK_RELEASE,
       KEY1_LONG_PRESS,
       KEY1_LONG_PRESS_CONTINUOUS,
       KEY1_LONG_PRESS_RELEASE,           //5       KEY2_CLICK,                                                   //6       KEY2_CLICK_RELEASE,
       KEY2_LONG_PRESS,
       KEY2_LONG_PRESS_CONTINUOUS,
       KEY2_LONG_PRESS_RELEASE,
       KEY3_CLICK,                                            //11       KEY3_CLICK_RELEASE,
       KEY3_LONG_PRESS,
       KEY3_LONG_PRESS_CONTINUOUS,
       KEY3_LONG_PRESS_RELEASE,
       KEY4_CLICK,                                     //16       KEY4_CLICK_RELEASE,
       KEY4_LONG_PRESS,
       KEY4_LONG_PRESS_CONTINUOUS,
       KEY4_LONG_PRESS_RELEASE,
       KEY5_CLICK,                                     //21       KEY5_CLICK_RELEASE,
       KEY5_LONG_PRESS,
       KEY5_LONG_PRESS_CONTINUOUS,
       KEY5_LONG_PRESS_RELEASE,
       KEY6_CLICK,                                     //26       KEY6_CLICK_RELEASE,
       KEY6_LONG_PRESS,
       KEY6_LONG_PRESS_CONTINUOUS,
       KEY6_LONG_PRESS_RELEASE,}KEY_VALUE_TYPEDEF;

我们这个项目总共有6个按键,每个按键需要检测短按、短按释放、长按、长按释放、连续长按5个功能,所以总共有30个不同的枚举值分别来对应不同按键的不同功能。

第二步:

自定义了函数指针类型以后,我们就可以通过KeyEvent_CallBack_t这个类型名称,去定义我们的函数指针变量。

KeyEvent_CallBack_t KeyScanCBS;

那KeyScanCBS就是函数指针,所以它的返回值是void类型,形参是KEY_VALUE_TYPEDEF枚举类型的。

最终就是把这个指针指向别的.c文件的函数,从而实现不同.c文件之间的数据传递,同时又能保持很好的可移植性(相互独立,互不干扰)。

那怎么指向呢?我的方法是重新定义一个函数,专门来为这个指针指向,这样方便别的.c文件调用,这个函数我称为注册函数

比如以下函数:

void hal_KeyScanCBSRegister(KeyEvent_CallBack_t pCBS){
       if(KeyScanCBS == 0)
       {
                     KeyScanCBS = pCBS;
       }}

这个函数的作用就是把我们前面定义的KeyScanCBS函数指针指向外部的函数地址(也就是要指向那个函数的函数名)。

当然,这个函数不是必须的,只是我的思维和代码风格,你也可以不单独写这样的函数,只要用之前把KeyScanCBS指向外部函数就可以了,否则等着程序死机吧哈哈哈。

第三步:

准备好这几步以后,我们继续来说下怎么去使用它。

我们哪里要用到按键的功能,就在那个.c文件那里重写一个同样的函数

比如说app.c这个文件是产品功能代码(应用层),我需要在应用层使用按键功能。

重写函数的时候,返回值和形参要跟那个函数指针类型一样。

如果你忘记了,那我们再来回顾下。

typedef void (*KeyEvent_CallBack_t)(KEY_VALUE_TYPEDEF keys);

无返回值,形参为KEY_VALUE_TYPEDEF类型。

只有这样,你才能把这个函数的地址赋值给KeyScanCBS这个指针,才能正常传递数据。

重写的这个函数就是通过形参来接收硬件层按键值的,如果是串口数据,也是同理,只是形参不一样。

然后,我们在产品功能初始化的函数里直接调用刚刚hal_key.c的注册函数

把KeyEventHandle这个函数的地址赋值给hal_key.c的KeyScanCBS这个函数指针。

所以,最终KeyScanCBS可以理解成等同于KeyEventHandle函数

我们在hal_key.c文件里,看按键检测解析程序,最终就是执行KeyScanCBS把我们keys(按键值)传递到我们app.c文件的。

这样,就能做到以事件去驱动,只有按键按下,并且真实有效,我才会调用KeyScanCBS,才会把按键值传递给应用层。

而中间,两个文件之间没有任何全局变量的依赖,也完全可以独立,大家可以细品消化一下。

这里有个细节就是为什么我函数的形参要用枚举类型。

如果你对接过一些模块(WiFi、蓝牙等)二次开发就知道了,模块核心代码都是封装成lib这种库给你的,你并看不到源代码。

只能用他们的函数,如果不用枚举,那你不知道形参可以传入什么值对吧?

如果用枚举,我把能用的值都列出来给你,并且起好名字,让你一看就知道是啥意思,这是不是就很方便?

Ok,今天就写到这里,大家下去可以做下实验。


推荐阅读

史海拾趣

洲光源(Chau Light)公司的发展小趣事

在洲光源公司的发展历程中,创新一直是其前进的动力源泉。公司始终坚持以市场需求为导向,不断进行技术创新和产品升级。通过与高校、科研机构的合作,洲光源成功引进了一批高端人才和先进技术,为公司的发展注入了新的活力。同时,公司还积极参与国家和地方的创新计划项目,不断推动科技创新和产业升级。这些创新举措不仅提升了洲光源公司的市场竞争力,也为公司的持续发展奠定了坚实的基础。

海芯科技(AVIA)公司的发展小趣事

面对电子行业的快速变化和市场竞争的加剧,海芯科技始终保持着对技术创新的追求和投入。公司不断引进新技术、新工艺和新材料,对现有产品进行升级和改进,同时也在不断探索和研发新的产品和技术。这些技术升级和创新发展不仅提升了公司的核心竞争力,也为公司在未来市场竞争中保持领先地位提供了有力保障。

这五个故事展示了海芯科技在电子行业中的发展历程和取得的成就。通过不断的努力和创新,海芯科技已经逐渐成为了电子行业中的佼佼者,为行业的发展做出了积极的贡献。

Hengstler GmbH公司的发展小趣事

海芯科技一直将产品研发视为公司发展的核心动力。经过长时间的研发与试验,公司成功推出了一系列具有市场竞争力的产品。这些产品不仅性能稳定、质量可靠,而且具有广泛的应用领域。随着产品在市场上的逐步推广和应用,海芯科技逐渐获得了客户的认可和信赖,市场份额也逐步扩大。

Frequency Electronics Inc公司的发展小趣事

为了进一步扩大市场份额,高频电子积极实施全球化战略。公司在全球范围内建立了销售网络和服务体系,与众多国际知名企业建立了长期合作关系。同时,高频电子还针对不同地区的市场需求,定制化开发符合当地标准的产品和服务。这些努力使得高频电子的产品和服务能够覆盖全球多个国家和地区,为公司带来了稳定的收入来源和持续增长的动力。

Amecon Magnetics公司的发展小趣事

在竞争激烈的电子行业中,品质是企业生存和发展的关键。Amecon Magnetics公司始终将品质管理放在首位,建立了严格的质量管理体系,确保从原材料采购到产品出厂的每一个环节都符合高标准的质量要求。这种对品质的执着追求使得公司的产品赢得了客户的广泛好评和信赖,为公司树立了良好的品牌形象。

3L Electronic Corporation公司的发展小趣事

3L Electronic Corporation,自XXXX年在台北创立以来,凭借创始人的远见卓识和团队的努力,逐渐在电子行业崭露头角。初期,公司主要生产电子零组件,凭借着精湛的工艺和稳定的质量,赢得了客户的信赖。随着市场的扩大,公司逐渐拓展到电子产品修理和国际贸易等领域,为后续的快速发展奠定了坚实基础。

问答坊 | AI 解惑

数字 PID及其改进算法

数字 PID及其改进算法…

查看全部问答>

初学者问几个问题

买 了个moto 的 linux手机,想做些小开发,没有linux下的开发经验,只做个windows下的开发,有些问题不明白,望各位指导下啊 1、交叉编译环境建立,网上看的一些资料,都 是说要下载这样那样的源码,然后编译,能不能不直接用别人编译好的?总不可 ...…

查看全部问答>

玩转CVSNT+TortoiseCVS 版本控制系统(图解)

玩转CVSNT+TortoiseCVS 版本控制系统(图解)…

查看全部问答>

DSP芯片的分类

DSP的芯片可以按照以下的三种方式进行分类。 1、按基础特性分       这是根据DSP芯片的工作时钟和指令类型来分类的。如果DSP芯片在某时钟频率范围内的任何频率上能正常工作,除计算速度有变化外,没有性能的下降,这类DSP芯 ...…

查看全部问答>

关于梅托勒称重方面问题

请教个问题 我们用梅托勒的称重终端 进行称重 现在 出现问题是 称不准 进行校正放20KG的砝码 显示正确的 但放2个20KG的 就是42或者没有 放3个就是61左右 反正就是总有误差放1000KG显示误差有4KG 我们是用3个传感器 通过接线盒接到仪表 传感器灵 ...…

查看全部问答>

【200分】中断驱动+应用层同步响应的问题咨询

问题描述: 1.一个流式驱动,工作流程是硬件触发一个中断,系统采集到数据信息后通过xxx_READ将采集到的数据发给应用层; 2.驱动通过一个线程函数来获取GPIO的中断信号,并读取值从打印信息打印出。 (这里的处理正常,有多少中断都接受到,并且 ...…

查看全部问答>

串口 3G

现在在做3g上网,改Com7 注册表改后必须reset RasEnumDevices 才能枚举出来修改后的信息,请问有没有什麽办法 改完注册表不需要重新启动啊?Thanks! …

查看全部问答>

请教嵌入式系统的两个简单问题

初学者,大家别见笑. 问题1:很多嵌入式系统启动时,其信息可以通过串口在另一台电脑的超级中断       里现实出来,请问嵌入式系统需要如何设置,它的信息可以输出到串口? 问题2:嵌入式系统的程序,很多是无限循环结构,如   ...…

查看全部问答>

51单片机。。。。

有 哪位高 手 能 告诉我,我把 自己编的 程序烧写到 自己 的 单片机中 却 没有 反映?这 是 怎么 回事? …

查看全部问答>