位置: 主页 > 数码 > 内容页
函数指针有啥用?有啥意义?

无际单片机编程   2023-08-04 11:21:17

很久没写烧脑文了,今天来写一篇。

说来奇怪,昨晚睡觉前,突然在想一个问题:函数指针有啥用?有啥意义?


(相关资料图)

起源是之前有个学员问我这个问题,但是感觉当时回答得不是特别好。

有些东西就是这样,自己知道该怎么用,用在哪,但如果一下子让我很通俗易懂地表达出来,脑子就卡壳了。

昨晚夜深人静,我想到了正好我们wifi报警主机那个项目有个例子,非常适合去深刻理解。

这个项目我们对接的是涂鸦云平台,然后涂鸦云提供SDK,我们需要移植到我们自己的项目里去。

在移植的过程中,我们需要去更改他们SDK里面的代码,我觉得这点是可以优化的。

下面给大家举一个具体的例子:

比如我们使用的涂鸦云提供的WiFi模组,通过串口和我们单片机连接,有一个通讯协议,但是涂鸦云帮我们做好了这个通讯协议的代码,所以会提供一个SDK给我们。

我们在使用SDK的时候,需要修改SDK的代码,把我们这款单片机的串口发送一个字节的函数,移植到SDK里面去。

Hal_Wifi_SendByte函数是STM32的串口发送一个字节函数。

我们要把这个函数放到涂鸦云SDK protocol.c文件的usart_transmit_output函数里面去。

最终SDK是调用usart_transmit_output函数实现串口数据流发送功能。

Ok,我们现在思考一下,这种方式有什么问题?如何解决?

问题1:

如果采用这种方式,有一种功能实现不了,就是假设涂鸦云这个SDK要封库(lib文件),源码不开放,我们就没法把Hal_Wifi_SendByte放到protocol.c文件的usart_transmit_output函数里面去。

是不是会有这个问题?这意味着,protocol.c必须开源,但是很多企业有一些核心代码,是商业机密,不能开放的,比如做导航的地图数据库。

问题2:

客户的技术水平层次不齐,如果有些基础稍微差点的,把SDK改乱了,这样会增加涂鸦云技术支持的人工成本。

所以,如果我去做,最理想的是不让客户改SDK的任何代码,你就直接按照我的方法和流程,调用函数用。

怎么解决这个问题?那就必须要用函数指针了!记住,是必须!

第一步:在protocol.h文件下自定一个函数指针类型pUart_transmit_output。

第二步:在protocol.c定义一个函数指针变量Uart_transmit_outputCBS。

第三步:修改protocol.c文件下uart_transmit_output函数,直接调用刚刚定义的函数指针变量Uart_transmit_outputCBS,记住调用前,必须要指针不为空的条件判断,否则,如果指针调用前没正确指向,会造成指针异常,程序死机。

第四步:在protocol.c文件下编写Uart_transmit_outputCBS函数指针的注册函数,也就是让这个函数指针变量指向一个地址的函数接口。

为什么要单独写个函数呢?因为我们理想的情况下,是提供函数接口,给别的.c文件调用,而不是用全局变量的形式。

第五步:别忘记在protocol.h文件下声明下Uart_transmit_outputCBSRegister函数,否则别的.c文件无法调用这个函数。

第六步:在我们单片机串口驱动文件hal_usart.c的初始化函数hal_UsartInit里面调用Uart_transmit_outputCBSRegister函数,然后把hal_Wifi_SendByte函数地址作为形参传递进去。

当然,在此之前别忘记在hal_usart.c文件的开头include SDK相关头文件。

最终分析:这样操作完以后,结果会是怎样?

Protocol.c文件下,调用uart_transmit_outputCBS函数指针时,是不是等同于调用hal_usart.c的hal_Wifi_SendByte函数?

这就是函数指针的作用,这样就能实现,客户不修改SDK代码,SDK也能调用客户工程下.c文件里的函数了。

这些技巧不复杂,就像纱窗,有人指点一下,很快就捅破了,没人指点,可能很多年都领悟不了,这也是进阶架构师必须要掌握的技巧。

审核编辑:汤梓红

更多智能

更多电脑