回调函数引发的自省

2024-03-08 21:38
文章标签 函数 引发 自省 回调

本文主要是介绍回调函数引发的自省,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

每次看到回调函数我都会纠结一下,因为我不会。尽管每次都会重新学习一遍,但是从来没用过,过几天就会忘记。经常看到别人说回调函数好用,今天就再次从网上学习了一遍回调函数。看完之后,觉得回调函数...就这?附段学习网站的代码:

#include<stdio.h>int Callback_1() // Callback Function 1
{printf("Hello, this is Callback_1 ");return 0;
}int Callback_2() // Callback Function 2
{printf("Hello, this is Callback_2 ");return 0;
}int Callback_3() // Callback Function 3
{printf("Hello, this is Callback_3 ");return 0;
}int Handle(int (*Callback)())
{printf("Entering Handle Function. ");Callback();printf("Leaving Handle Function. ");
}int main()
{printf("Entering Main Function. ");Handle(Callback_1);Handle(Callback_2);Handle(Callback_3);printf("Leaving Main Function. ");return 0;
}

运行结果如下:

Entering Main Function.
Entering Handle Function.
Hello, this is Callback_1
Leaving Handle Function.
Entering Handle Function.
Hello, this is Callback_2
Leaving Handle Function.
Entering Handle Function.
Hello, this is Callback_3
Leaving Handle Function.
Leaving Main Function.

看完上面的代码,我的第一感觉,就是回调函数多此一举,为什么不在主函数中直接调用三个callback函数呢?这样不是更加直接有效吗?后来总觉得回调函数的作用不应该这么简单,于是我再次搜索相关的资料,得到如下结果(关于回调函数的异步机制):

以下内容取自:https://blog.csdn.net/xiabodan/article/details/47999411?utm_source=copy

我们首先假设有两个程序员在写代码,A程序员写底层驱动接口,B程序员写上层应用程序,然而此时底层驱动接口A有一个数据d需要传输给B,此时有两种方式:
   1、A将数据d存储好放在接口函数中,B自己想什么时候去读就什么时候去读,这就是我们经常使用的函数调用,此时主动权是B。
   2、A实现回调机制,当数据变化的时候才将通知B,你可以来读取数据了,然后B在用户层的回调函数中读取数据d,完成OK。此时主动权是A。
很明显第一种方法太低效了,B根本就不知道什么时候该去调用接口函数读取数据d。而第二种方式由于B的读取数据操作是依赖A的,只有A叫B读数据,那么B才能读数据。也即是实现了中断读取。
那么回调是怎么实现的呢,其实回调函数就是一个通过函数指针调用的函数。如果用户层B把函数的指针(地址)作为参数传递给底层驱动A,当这个指针在A中被用为调用它所指向的函数时,我们就说这是回调函数。
注意:是在A中被调用,这里看到尽管函数是在B中,但是B却不是自己调用这个函数,而是将这个函数的函数指针通过A的接口函数传自A中了,由A来操控执行,这就是回调的意义所在。
下面就通过一个例子来演示
首先写A程序员的代码

//-----------------------底层实现A-----------------------------
typedef void (*pcb)(int a); //函数指针定义,后面可以直接使用pcb,方便
typedef struct parameter{int a ;pcb callback;
}parameter; void* callback_thread(void *p1)//此处用的是一个线程
{//do somethingparameter* p = (parameter*)p1 ;while(1){printf("GetCallBack print! \n");sleep(3);//延时3秒执行callback函数p->callback(p->a);//函数指针执行函数,这个函数来自于应用层B}
}//留给应用层B的接口函数
extern SetCallBackFun(int a, pcb callback)
{printf("SetCallBackFun print! \n");parameter *p = malloc(sizeof(parameter)) ; p->a  = 10;p->callback = callback;//创建线程pthread_t thing1;pthread_create(&thing1,NULL,callback_thread,(void *) p);pthread_join(thing1,NULL);
}

上面的代码就是底层接口程序员A写的全部代码,留出接口函数SetCallBackFun即可

下面再实现应用者B的程序,B负责调用SetCallBackFun函数,以及增加一个函数,并将吃函数的函数指针通过SetCallBackFun(int a, pcb callback)的第二个参数pcb callback 传递下去。

//-----------------------应用者B-------------------------------
void fCallBack(int a)       // 应用者增加的函数,此函数会在A中被执行
{//do somethingprintf("a = %d\n",a);printf("fCallBack print! \n");
}int main(void)
{SetCallBackFun(4,fCallBack);return 0;
}

运行程序会看到

先会打印A程序的                 printf("GetCallBack print! \n");
然后等待3秒钟才会打印应用者B的    printf("fCallBack print! \n");

 

 从上面的内容来看,给我最大的体会就是:回调函数带来的好处就是主动权的问题。比如在单片机使用串口和4G模块进行通信的时候,再也不用一直循环或中断模式来等待4G模块上报URC了,我们可以使用回调函数来随时处理AT指令回复的信息。

但回调的优点仅仅如此吗?

通过在技术群的讨论,总结后我获得了一个新的关键词:解耦合。

程序设计经常提到的解耦,到底是要解除什么之间的耦合?我知乎了一下,有的人说解耦就是将程序积木化,各个积木可以组合在一起而形成一个形状,又可以拆分,又可以替换,因为基本上各个积木块都是独立的,只要他们直接的接口(形状)匹配,就可以灵活的组合在一起。当然,这是理想状态。解耦就是在逐渐达到这个理想状态。但有个人的说法让我醍醐灌顶:少用全局变量

我每次写代码的时候,写之前虽然会先设计框架,保证代码的耦合性,但是为了图方便,后期写代码的时候还是会慢慢的添加各种全局变量,因为不管是在linux的多线程中还是freertos/ucos的任务中,全局变量都是共享的,可以直接拿来用(前提是保证同一时刻只有一个线程/任务修改全局变量)。随着全局变量越来越多,当程序出现bug的时候,问题就来了:牵一发而动全身。

总结:在实现产品功能的时候,务必也要注意代码的耦合性,写好代码,而不是写完代码。

 

 

这篇关于回调函数引发的自省的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/788474

相关文章

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、

golang panic 函数用法示例详解

《golangpanic函数用法示例详解》在Go语言中,panic用于触发不可恢复的错误,终止函数执行并逐层向上触发defer,最终若未被recover捕获,程序会崩溃,recover用于在def... 目录1. panic 的作用2. 基本用法3. recover 的使用规则4. 错误处理建议5. 常见错

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

Java function函数式接口的使用方法与实例

《Javafunction函数式接口的使用方法与实例》:本文主要介绍Javafunction函数式接口的使用方法与实例,函数式接口如一支未完成的诗篇,用Lambda表达式作韵脚,将代码的机械美感... 目录引言-当代码遇见诗性一、函数式接口的生物学解构1.1 函数式接口的基因密码1.2 六大核心接口的形态学

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

Python中的异步:async 和 await以及操作中的事件循环、回调和异常

《Python中的异步:async和await以及操作中的事件循环、回调和异常》在现代编程中,异步操作在处理I/O密集型任务时,可以显著提高程序的性能和响应速度,Python提供了asyn... 目录引言什么是异步操作?python 中的异步编程基础async 和 await 关键字asyncio 模块理论

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function