C/C++面向对象(OOP)编程-回调函数详解(回调函数、C/C++异步回调、函数指针)

本文主要介绍回调函数的使用,包括函数指针、异步回调编程、主要通过详细的例子来指导在异步编程和事件编程中如何使用回调函数来实现。

🎬个人简介:一个全栈工程师的升级之路!
📋个人专栏:C/C++精进之路
🎀CSDN主页 发狂的小花
🌄人生秘诀:学习的本质就是极致重复!

目录

1 回调函数

2 C语言回调函数

2.1 函数指针

2.1.1 函数指针变量

2.1.2 定义函数类型

2.1.3 定义函数指针类型

2.2 C语言回调函数实例

2.3 C语言异步编程

3 C++回调函数

3.1 C++简单的回调实现

3.2 简答的异步回调

3.3 静态成员函数实现异步回调(有线程)


1 回调函数

        回调函数是一种特殊的函数,它作为参数传递给另一个函数,并在被调用函数执行完毕后被调用。这种机制常用于事件处理、异步编程和处理各种操作系统和框架的API。

具体来说,一个函数可以将其它的函数(也就是回调函数)作为参数传入,当特定的事件触发或者任务完成时,这个被传入的函数就自动执行。这样的设计使得代码更加模块化,增强了代码的可读性和可重用性。

        回调函数是一种设计理念,与编程语言无关,C语言中可以通过函数指针实现回调函数

        回调函数主要用于异步编程和事件处理中。

        回调函数提供了一种强大的工具,允许我们将复杂的问题分解为更小的部分进行解决,同时也增加了代码的灵活性和扩展性。

2 C语言回调函数

        C语言的回调函数是使用函数指针实现的,函数作为参数传给另一个函数,传参的函数被称为回调函数。

2.1 函数指针

2.1.1 函数指针变量

        指向函数的指针变量称为函数指针。本质上是一个指针。

        定义方式:

        返回类型 (*指针变量名)(参数列表);

        其中,返回类型表示函数返回值的类型,参数列表表示函数接受的参数类型和个数。指针变量名就是函数指针的名称,它可以像普通变量一样被赋值和使用。

        使用实例:

  例如,定义一个指向返回值为int类型、接受两个int类型参数的函数的指针:

int (*pFunc)(int, int);

C++中可以使用new运算符动态分配内存来存储函数指针:

int (*pFunc)(int, int) = new int(*func)(int a, int b);

其中,func是一个已定义的函数,pFunc是指向该函数的指针。使用delete运算符释放内存:

delete pFunc;

2.1.2 定义函数类型

       定义方式:

        typedef 返回类型 新的函数名(参数列表);

        使用实例:

定义一个返回类型是int,参数为两个int数据,的函数类型

typedef int INT_func(int,int);

        调用方式:

int sub(int a, int b)
{return (a-b);
}// add 为指针类型,函数的地址,因此需要声明变量为指针类型
INT_func *add_p = add;
int ret = add_p(12,13);

一个例子:

#include <stdio.h>typedef int INT_func(int,int);int add(int a,int b)
{return (a+b);
}int sub(int a, int b)
{return (a-b);
}int multi(int a,int b)
{return (a*b);
}int main()
{INT_func *add_p = add;int ret = add_p(12,13);printf("12+13 = %d \n",ret);INT_func *sub_p = sub;int ret1 = sub_p(12,13);printf("12-13 = %d \n",ret1);INT_func *mutti_p = multi;int ret2 = mutti_p(12,23);printf("12*13 = %d \n",ret2);return 0;
}

        运行结果:

2.1.3 定义函数指针类型

         定义方式:

        typedef 返回类型 (*新的函数名)(参数列表);

        使用实例:

定义一个返回类型是int,参数为两个int数据,的函数类型

typedef int (*INT_func)(int,int);

        调用方式:

int sub(int a, int b)
{return (a-b);
}// add 为指针类型,函数的地址,因此需要声明变量为指针类型
INT_func  add_p = add;
int ret = add_p(12,13);

一个例子:

#include <stdio.h>typedef int (*INT_func)(int,int);int add(int a,int b)
{return (a+b);
}int sub(int a, int b)
{return (a-b);
}int multi(int a,int b)
{return (a*b);
}int main()
{INT_func add_p = add;int ret = add_p(12,13);printf("12+13 = %d \n",ret);INT_func sub_p = sub;int ret1 = sub_p(12,13);printf("12-13 = %d \n",ret1);INT_func mutti_p = multi;int ret2 = mutti_p(12,23);printf("12*13 = %d \n",ret2);return 0;
}

        运行结果:

2.2 C语言回调函数实例

        在C语言中,回调函数是一种常见的编程模式,它允许一个函数作为参数传递给另一个函数。这种特性使得我们可以将一些特定的任务委托给其他函数来完成。

#include <stdio.h>// 定义一个函数指针类型
typedef void (*Callback)(int);// 定义一个函数,接受一个回调函数作为参数
void do_something(Callback callback) {// 在这里做一些操作...printf("Doing something...\n");// 然后调用回调函数callback(42);
}// 定义一个回调函数
void my_callback(int data) {printf("Callback called with data: %d\n", data);
}int main() {// 将回调函数传递给do_something函数do_something(my_callback);return 0;
}

        在这个例子中,do_something函数接受一个Callback类型的参数,这是一个指向函数的指针。然后,do_something函数调用这个回调函数,并将一个整数作为参数传递给它。

my_callback函数是一个简单的回调函数,它接受一个整数参数并打印出来。

main函数中,我们将my_callback函数传递给do_something函数,这样当do_something函数完成其操作后,它会调用my_callback函数。

        运行结果:

2.3 C语言异步编程

        回调函数在异步编程中有着重要的作用,在接口编程应用广泛,实战中经常会注册一个回调函数来接收其他程序返回的数据,这样做的好处是调用者无需主动去获取被调用者的数据,仅仅需要回调,让被调用者去传送数据,这样就形成了回调,这是回调最常见的应用。

        假设A是视频流数据,每隔一段时间会将数据传送给B,B有两种接收方式:

        (1)A将视频数据存储好,提供接口,B根据自己的需求去调用,无需考虑时间,主动权在B。

        (2)A利用回调机制,实现回调函数,当数据来临时通知B来取数据,B注册回调函数接收数据,这样主动权在A。

        很显然第一种取数据的方式,由于B不知道数据何时来临比较低效,而第二种取数据方式A通知B来取数据,效率比较高,间接的实现了中断取数。

        这里就用到了回调,B注册一个函数,将函数地址传给A,A中调用该地址获取到B注册的函数时,我们就称这个函数为回调函数。

一个例子:

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <unistd.h>// A的实现,一般会隐藏
typedef void (*CallbackPtr)(int); //函数指针定义typedef struct parameter{int a ;CallbackPtr callback;
}parameter; // 创建实例
parameter ParaData;void* callback_thread(void *p1)//此处用的是一个线程
{//do something// 循环改变p->a的值为为0 2 3 4 5 6 7 8 9 ,每隔3秒改变一次parameter* p = (parameter*)p1 ;while(1){sleep(3);//延时3秒执行callback函数p->callback(p->a);//函数指针执行函数,这个函数来自于应用层Bp->a = (p->a + 1) % 10;}
}void startup_app_A()
{// 初始化为0ParaData.a = 0;CallbackPtr init_call_back;ParaData.callback = init_call_back;//创建线程pthread_t thing1;pthread_create(&thing1,NULL,callback_thread,(void *) &ParaData);
}// 给B的接口,接收注册函数
extern void SetCallBackFun(CallbackPtr callback)
{printf("SetCallBackFun print! \n");ParaData.callback = callback;
}// //-----------------------应用者B-------------------------------
void fCallBack(int a)       // 应用者增加的函数,此函数会在A中被执行
{//do somethingprintf("B得到A的数据 = %d\n",a);
}int main(void)
{// 启动Astartup_app_A();SetCallBackFun(fCallBack);// 主函数while (1){// std::cout << "main function" << std::endl;printf("main function\n");sleep(2);}return 0;
}

        运行结果:

        分析结果,可以看出A提供的数据会回调B,B会每隔一定的时间收到数据,完成回调,对于一般的编程会将A的操作封装起来,只提供以下接口和函数指针类型:

typedef void (*CallbackPtr)(int);
extern void SetCallBackFun(CallbackPtr callback);

        这样B可以每次注册回调函数获取A传来的数据,如下:

void fCallBack(int a)       // 应用者增加的函数,此函数会在A中被执行
{//do somethingprintf("B得到A的数据 = %d\n",a);
}

        主函数中调用:

SetCallBackFun(fCallBack);

        这样就完成了一个C语言回调的异步编程,这在驱动领域极其常见,比如摄像头输出50fps的UVC流,程序员用UVC来完成相应的任务,这样就可以使用异步回调的方式实现。

3 C++回调函数

        C++回调函数是一种将函数作为参数传递给另一个函数的技术。在C++中,回调函数通常使用函数指针或std::function对象来实现。

        函数回调类型的定义:

typedef std::function<void(int)> Callback;

        上述定义了一个返回值为void,参数为一个int类型参数的函数回调类型Callback,等同于以下的函数指针类型:

typedef void (*Callback)(int);

3.1 C++简单的回调实现

#include <iostream>
#include <functional>// 定义一个回调函数类型
typedef std::function<void(int)> Callback;// 定义一个接受回调函数的函数
void process(int value, Callback callback) {std::cout << "处理值: " << value << std::endl;callback(value); // 调用回调函数
}// 定义一个回调函数
void myCallback(int result) {std::cout << "回调结果: " << result << std::endl;
}int main() {int value = 42;process(value, myCallback); // 传递回调函数给process函数return 0;
}

        运行结果:

3.2 简答的异步回调

#include <iostream>
#include <functional>
#include <thread>
#include <chrono>
#include <unistd.h>
void async_operation(std::function<void(int)> callback) {// 模拟一些计算std::this_thread::sleep_for(std::chrono::seconds(1));int result = 42;// 调用回调函数并将结果传递给它callback(result);
}
void callback_function(int result) {std::cout << "Async operation completed with result: " << result << std::endl;
}
int main() {// 发起异步操作,并在操作完成后调用回调函数async_operation(callback_function);// 主线程继续执行其他任务while (1){std::cout << "Main thread continues to do other work..." << std::endl;sleep(1);}return 0;
}

        运行结果:

3.3 静态成员函数实现异步回调(有线程)

        当有线程时,非静态成员函数无法实现函数的回调。需要使用静态成员函数进行回调。

#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <unistd.h>
using namespace std;//--------------
// 类中定义线程,并实现回调
// A应用
template <typename Tobject, typename Tparam>
class AsynA
{public:struct ThreadParam{AsynA *pthis;Tparam a;}; // 根据线程参数自定义结构typedef void (Tobject::*Cbfun)(Tparam);public:AsynA() : number(1){printf("AsynA : AsynA()\n");}void print(void){printf("print : number = %d \n", number);}static void *funCallbackThread(void *param); void CreateCallbackThread(Tobject *pInstance, Cbfun pFun, Tparam a);void ThreadFunc(Tparam a);                   private:int number;Cbfun pCbfun;Tobject *m_pInstance;
};template <typename Tobject, typename Tparam>
void AsynA<Tobject, Tparam>::CreateCallbackThread(Tobject *pInstance, Cbfun pFun, Tparam a)
{ThreadParam *p = new ThreadParam;p->pthis = this;p->a = a;m_pInstance = pInstance;pCbfun = pFun;pthread_t thing1;pthread_create(&thing1, NULL, &funCallbackThread, (void *)p);
}template <typename Tobject, typename Tparam>
void *AsynA<Tobject, Tparam>::funCallbackThread(void *param)
{ThreadParam *p = (ThreadParam *)param;printf("ThreadParam p->a = %d\n", p->a);p->pthis->ThreadFunc(p->a);return 0;
}
template <typename Tobject, typename Tparam>
void AsynA<Tobject, Tparam>::ThreadFunc(Tparam a)
{printf("%d ThreadFunc : number = %d \n",number);while (1){const pthread_t threadID = pthread_self();sleep(a);(m_pInstance->*pCbfun)((int)threadID);}
}
// B应用
class AsynB
{public:AsynB() : number(0){}void fCallBack(int p){printf("Thread ID = %ld\n", p);}private:int number;
};int main(void)
{// 类中定义线程,并实现回调AsynA<AsynB, int> asyn_a;AsynB asyn_b;int p = 2;asyn_a.CreateCallbackThread(&asyn_b, &AsynB::fCallBack, p);AsynA<AsynB, int> asyn_a1;AsynB asyn_b1;int p1 = 4;asyn_a1.CreateCallbackThread(&asyn_b1, &AsynB::fCallBack, p1);// 主程序阻塞在这里getchar();return 0;
}

        运行结果:

        分析可知,上述代码完成在类中使用静态成员函数的异步回调,创建了两个子线程。

🌈我的分享也就到此结束啦🌈
如果我的分享也能对你有帮助,那就太好了!
若有不足,还请大家多多指正,我们一起学习交流!
📢未来的富豪们:点赞👍→收藏⭐→关注🔍,如果能评论下就太惊喜了!
感谢大家的观看和支持!最后,☺祝愿大家每天有钱赚!!!

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

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

相关文章

Spark魔力:招聘网站数据深度分析系统

Spark魔力&#xff1a;招聘网站数据深度分析系统 简介数据集技术栈功能特点创新点 简介 在本文中&#xff0c;我们将介绍一款基于Spark的招聘网站数据分析系统&#xff0c;该系统使用爬取的前程无忧招聘数据。通过结合Flask、Pandas、PySpark、以及MySQL等技术&#xff0c;实现…

【汇编笔记】初识汇编-内存读写

汇编语言的由来&#xff1a; CPU是计算机的核心&#xff0c;由于计算机只认识二进制&#xff0c;所以CPU执行的指令是二进制。 我们要想让CPU工作&#xff0c;就得给他提供它认识的指令&#xff0c;这一系列的指令的集合&#xff0c;称之为指令集。 指令集&#xff1a; 不同的体…

单片机键盘程序设计举例

1、键盘与的连接 图3键盘连接 图4单片机与键盘接口图 2、通过1/0口连接。将每个按钮的一端接到单片机的I/O口&#xff0c;另一端接地&#xff0c;这是最简单的办法&#xff0c;如图3所示是实验板上按钮的接法&#xff0c;四个按钮分别接到P3.2 、P3.3、P3.4和P3.5。对于这种键…

css 设置鼠标覆盖显示菜单

鼠标覆盖到“全部分类”效果如下 鼠标放到“精品推荐”效果如下 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"&g…

【CF比赛记录】—— Good Bye 2023(A、B、C)

&#x1f30f;博客主页&#xff1a;PH_modest的博客主页 &#x1f6a9;当前专栏&#xff1a;CF比赛记录 &#x1f48c;其他专栏&#xff1a; &#x1f534;每日一题 &#x1f7e1; cf闯关练习 &#x1f7e2; C语言跬步积累 &#x1f308;座右铭&#xff1a;广积粮&#xff0c;缓…

mysql基础-表操作

环境&#xff1a; 管理工具&#xff1a;Navicat 数据库版本&#xff1a;5.7.37 mysql的版本&#xff0c;我们可以通过函数&#xff0c;version()进行查看&#xff0c;本次使用的版本如下&#xff1a; 目录 1.管理工具 1.1创建表 1.2.修改表名 1.3.复制表 1.4.删除表 2…

C#上位机与欧姆龙PLC的通信08----开发自己的通讯库读写数据

1、介绍 前面已经完成了7项工作&#xff1a; C#上位机与欧姆龙PLC的通信01----项目背景-CSDN博客 C#上位机与欧姆龙PLC的通信02----搭建仿真环境-CSDN博客 C#上位机与欧姆龙PLC的通信03----创建项目工程-CSDN博客 C#上位机与欧姆龙PLC的通信04---- 欧姆龙plc的存储区 C#上…

Linux常用命令大全总结及讲解(超详细版)

前言&#xff1a; Linux 是一个基于Linux 内核的开源类Unix 操作系统&#xff0c;Linus Torvalds于 1991 年 9 月 17 日首次发布的操作系统内核。Linux 通常打包为Linux 发行版。 Linux 最初是为基于Intel x86架构的个人计算机开发的&#xff0c;但此后被移植到的平台比任何其…

前端实现websocket类封装

随着Web应用程序的发展&#xff0c;越来越多的人开始利用Websocket技术来构建实时应用程序。Websocket是一种在客户端和服务器之间建立持久连接的协议。这种协议可以在一个单独的连接上实现双向通信。与HTTP请求-响应模型不同&#xff0c;Websocket允许服务器自主地向客户端发送…

OpenCV-Python(21):OpenCV中的轮廓性质

3.轮廓的性质 本文我们将主要学习基于轮廓来提取一些经常使用的对象特征。 3.1 长宽比 边界矩形的宽高比&#xff1a; x,y,w,h cv2.boundingRect(cnt) aspect_ratio float(w)/h 3.2 Extent 轮廓面积与边界矩形面积的比。 area cv2.contourArea(cnt) x,y,w,h cv2.bounding…

基于JavaWeb实验室预约管理系统(源码+数据库+文档)

一、项目简介 本项目是一套基于JavaWeb实验室预约管理系统&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Java学习者。 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;e…

simulink代码生成(五)——ePWM模块初级应用

前面分别讲到了SCI及ADC的配置及使用&#xff0c;现在梳理一下ePWM的配置和使用&#xff1b; 先打一些基础的DSP28335的基础知识&#xff1b; F28335 关于ePWM中断与SOC采样信号的一些思考_socasel-CSDN博客 F28335 ePWM模块简介——TMS320F28335学习笔记&#xff08;四&…

MySQL数据库索引优化

一、引言 1. 索引的重要性 MySQL数据库索引的重要性主要体现在&#xff0c;一是查询速度优化&#xff0c;索引可以极大地提高查询速度。对于没有索引的表&#xff0c;MySQL必须进行全部扫描来找到所需的行&#xff0c;如果表中数据量很大&#xff0c;那么通常很慢。通过适当的…

iPhone 13 Pro 更换『移植电芯』和『超容电池』体验

文章目录 考虑换电池Ⅰ 方案一Ⅱ 方案二 总结危险 Note系列地址 简 述: 首发买的iPhone 13P &#xff08;2021.09&#xff09;&#xff0c;随性使用一年出头&#xff0c;容量就暴跌 85%&#xff0c;对比朋友一起买的同款&#xff0c;还是95%。这已经基本得一天两充 >_<&a…

数据库进阶教学——读写分离(Mycat1.6+Ubuntu22.04主+Win10从)

目录 1、概述 2、环境准备 3、读写分离实验 3.1、安装jdk 3.2、安装Mycat 3.3、配置Mycat 3.3.1、配置schema.xml ​​​​3.3.2、配置server.xml 3.4、修改主从机远程登陆权限 3.4.1、主机 3.4.2、从机 3.5、启动Mycat 3.6、登录Mycat 3.7、验证 1、概述 读写分…

模型量化之AWQ和GPTQ

什么是模型量化 模型量化&#xff08;Model Quantization&#xff09;是一种通过减少模型参数表示的位数来降低模型计算和存储开销的技术。一般来说&#xff0c;模型参数在深度学习模型中以浮点数&#xff08;例如32位浮点数&#xff09;的形式存储&#xff0c;而模型量化可以…

OpenCV-Python(21):轮廓特征及周长、面积凸包检测和形状近似

2. 轮廓特征 轮廓特征是指由轮廓形状和结构衍生出来的一些特征参数。这些特征参数可以用于图像识别、目标检测和形状分析等应用中。常见的轮廓特征包括&#xff1a; 面积&#xff1a;轮廓所包围的区域的面积。周长&#xff1a;轮廓的周长&#xff0c;即轮廓线的长度。弧长&…

Docker自建文件快递柜系统

Docker自建文件快递柜系统。 软件特色&#xff1a; 轻量简洁&#xff1a;FastapiSqlite3Vue2ElementUI 轻松上传&#xff1a;复制粘贴&#xff0c;拖拽选择 多种类型&#xff1a;文本&#xff0c;文件 防止爆破&#xff1a;错误次数限制 防止滥用&#xff1a;IP限制上传次数…

GO语言基础笔记(七):网络编程

目录 Go语言网络协议基础 协议 实现 跨平台网络抽象 简单代码展示 服务端 客户端 服务端客户端通信实战 Go Linux服务端 Go Linux客户端 Windows C 客户端 总结 Go语言网络协议基础 在 Go 语言中&#xff0c;net/http 包提供了强大的工具来创建 HTTP 服务器。…

新产品推广选品牌外包广州迅腾文化传播多渠道传播能力

在当今激烈的市场竞争中&#xff0c;新产品推广已成为企业发展的关键。选择具备多渠道传播能力的品牌外包服务提供商&#xff0c;有助于快速提升品牌知名度和市场占有率。作为行业领先者&#xff0c;迅腾文化凭借卓越的多渠道传播能力&#xff0c;成为企业新产品推广的理想合作…