回调函数使用详解_回调函数的用法-程序员宅基地

技术标签: 回调函数  C语言知识总结  

转载自:https://blog.csdn.net/miao19920101/article/details/75648491

回调函数的使用

回调函数在C语言中是通过函数指针来实现的,通过将回调函数的地址传给被调函数从而实现回调。因此,要实现回调,必须首先定义函数指针。

1. 回调指针
概念:指针是一个变量,是用来指向内存地址的。一个程序运行时,所有和运行相关的物件都是需要加载到内存中,这就决定了程序运行时的任何物件都可以用指针来指向它。函数是存放在内存代码区域内的,它们同样有地址,因此同样可以用指针来存取函数,把这种指向函数入口地址的指针称为函数指针。

1. 采用函数调用的一般形式
首先看一个hello world!的程序:

int application_start( void )
{
     OSStatus err = kNoErr;
     char *s ="hello world !";
     app_log(" s:%s",s);
     return err;
}


打印的结果是:

[0][TCP: main.c:  90]  s:hello world !


如果采用函数调用的形式来实现:

//声明
void Islog( char *s);

int application_start( void )
{
     OSStatus err = kNoErr;
     Islog("hello world !");
     return err;
}

void Islog( char *s){
     app_log(" s:%s",s);
}


打印的结果是:

[0][TCP: main.c:  90]  s:hello world !


2. 简单的函数指针的应用
形式1:返回类型(*函数名)(参数表)

把上面的例子改成使用简单的函数指针的写法:

//声明
void Islog( char *s);

int application_start( void )
{
     OSStatus err = kNoErr;
     void (*fp)(char *s);//声明一个函数指针(fp)
     fp = Islog;//将Islog的函数入口地址付给fp
     fp("hello world !");//函数指针fp实现函数调用
     return err;
}

void Islog( char *s){
     app_log(" s:%s",s);
}


打印的结果是:

[0][TCP: main.c:  90]  s:hello world !


由上知道:函数指针函数的声明之间唯一区别就是:用指针名(*fp)代替了函数名Islog,这样声明了一个函数指针,然后进行赋值fp= Islog 就可以进行指针函数的调用了,声明函数指针时,只要返回值类型、参数个数、参数类型等保持一致,就可以声明一个函数指针了。注意,函数指针必须括号括起来 void (*fp)(char *s)。

3. 使用typedef更简单
实际中,为了方便,通常使用宏定义的方式声明函数指针。

形式2:返回类型(*新类型)(参数表)

typedef void (*intFunc)(int);
此宏定义的意思是要定义的类型是void (*)(int),即参数一个int,什么也不返回的函数指针,定义的别名是intFunc。

采用宏定义的方式声明函数指针将上上面的代码改写:

//宏定义
typedef void(*FP)(char *s);
//声明
void Islog( char *s);

int application_start( void )
{
     OSStatus err = kNoErr;
     FP fp; //通常使用宏FP来声明一个函数指针fp
     fp = Islog;//将Islog的函数入口地址付给fp
     fp("hello world !");//函数指针fp实现函数调用
     return err;
}

void Islog( char *s){
     app_log(" s:%s",s);
}


打印的结果是:

[0][TCP: main.c:  90]  s:hello world !


4. 函数指针数组
下面的是指针函数数组的例子:

例子1:

//宏定义
typedef void(*FP)(char *s);
//声明
void Islog1( char *s);
void Islog2( char *s);
void Islog3( char *s);

int application_start( void )
{
     OSStatus err = kNoErr;

  //void* Islog[] = {Islog1,Islog2,Islog3};//定义了指针数组,这里a是一个普通指针
  //Islog3[0] ("hello world !");//编译错误,指针数组不能用下标的方式来调用函数

     FP Islog[] = {Islog1,Islog2,Islog3};//定义一个函数指针的数组,这里的f是一个函数指针
     Islog[0] ("hello world!");
     Islog[1] ("hello world!");
     Islog[2] ("hello world!");
     return err;
}

void Islog1( char *s){app_log(" s:%s",s);}
void Islog2( char *s){app_log(" s:%s",s);}
void Islog3( char *s){app_log(" s:%s",s);}
打印的结果是:

[0][TCP: main.c: 101]  s:hello world!
[4][TCP: main.c: 103]  s:hello world!
[7][TCP: main.c: 105]  s:hello world!


例子2:

//宏定义
typedef void(*FP)( char *s,int count);
//声明
void Islog1( char *s,int count);
void Islog2( char *s,int count);
void Islog3( char *s,int count);

int application_start( void )
{
     OSStatus err = kNoErr;

  //void* Islog[] = {Islog1,Islog2,Islog3};//定义了指针数组,这里a是一个普通指针
  //Islog3[0] ("hello world !");//编译错误,指针数组不能用下标的方式来调用函数

     FP Islog[] = {Islog1,Islog2,Islog3};//定义一个函数指针的数组,这里的f是一个函数指针
     Islog[0] ("hello world!",1);
     Islog[1] ("hello world!",2);
     Islog[2] ("hello world!",3);
     return err;
}

void Islog1( char *s,int count){app_log(" s:%s and count:%d",s,count);}
void Islog2( char *s,int count){app_log(" s:%s and count:%d",s,count);}
void Islog3( char *s,int count){app_log(" s:%s and count:%d",s,count);}
打印的结果是:

[0][TCP: main.c: 101]  s:hello world! and count:1
[5][TCP: main.c: 102]  s:hello world! and count:2
[9][TCP: main.c: 103]  s:hello world! and count:3


2. 回调函数
概念:回调函数,顾名思义,就是使用者自己定义一个函数,使用者自己实现这个函数的程序内容,然后把这个函数作为参数传入别人(或系统)的函数中,由别人(或系统)的函数在运行时来调用的函数。函数是你实现的,但由别人(或系统)的函数在运行时通过参数传递的方式调用,这就是所谓的回调函数。简单来说,就是由别人的函数运行期间来回调你实现的函数。再来看看来自Stack Overflow某位大神简洁明了的表述:A “callback” is any function that is called by another function which takes the first function as a parameter。 也就是说,函数 FuncA 调用函数 FuncB)的时候,函数 FuncA通过参数给 函数 FuncB传递了另外一个函数 FuncC 的指针,在函数 FuncB 执行的过程中,函数FuncB 调用了函数 FuncC,这个动作就叫做回调(Callback),而先被当做指针传入、后面又被回调的函数 F3 就是回调函数。到此应该明白回调函数的定义了吧?

我们将一开始的hello world函数修改成函数回调样式:

1. 简单的回调函数

//定义回调函数
void PrintText(){
     app_log("hello world!");
}

//定义实现回调函数的“调用函数”
void CallprintfText(void (*callfunc)()){
     callfunc();
}

int application_start( void )
{
     OSStatus err = kNoErr;
     CallprintfText(PrintText);
     return err;
}
打印的结果是:

[0][TCP: main.c:  82] hello world!


2. 使用typedef写法:

typedef void (*CallFunc)();
//定义回调函数

void PrintText(){
     app_log("hello world!");
}

//定义实现回调函数的“调用函数”
void CallprintfText(CallFunc callfunc){
     callfunc();
}

int application_start( void )
{
     OSStatus err = kNoErr;
     CallprintfText(PrintText);
     return err;
}   
打印的结果是:

[0][TCP: main.c:  82] hello world!


3. 修改带参数的回调函数写法

这里只放了一个类型的参数,很重要!

void PrintText(char *s){
     app_log("s:%s",s);
}

//定义实现回调函数的“调用函数”
void CallprintfText(void (*callfunc)(char*),char *s){
    callfunc(s);
}

int application_start( void )
{
     OSStatus err = kNoErr;
     CallprintfText(PrintText,"hello world!");
     return err;
}
打印的结果是:

[0][TCP: main.c:  82] hello world!


4. 使用typedef写法:
参数类型两种

typedef void (*CallFunc)(char *s,int count);
//定义回调函数

void PrintText(char *s,int count){
     app_log("s:%s",s);
     app_log("count:%d",count);
}

//定义实现回调函数的“调用函数”
void CallprintfText(CallFunc callfunc,char *s,int count){
     callfunc(s,count);
}

int application_start( void )
{
     OSStatus err = kNoErr;
     CallprintfText(PrintText,"hello world!",1);
     return err;
}

打印的结果是:

[0][TCP: main.c:  84] s:hello world!
[3][TCP: main.c:  85] count:1


 

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_31869107/article/details/103506658

智能推荐

android sql插入数据,Android:插入/更新多个记录到SQL数据库-程序员宅基地

文章浏览阅读533次。好吧,也许我完全错过了一些东西,但我刚刚开始使用SQL和Android,并且从我所看到的更新和插入到数据库需要单独处理?我可以检索包含多个记录的游标,但是如果我想将更新应用于游标中的每个记录,我只能看到如何单独使用ContentValues和Update(),然后光标需要重装?Android:插入/更新多个记录到SQL数据库同样如果我想创建多个新记录,这些似乎需要单独插入?我可以不创建列表和批量插..._android从数据库加载几万条记录到数据库

elasticsearch实现博客搜索_博客文章通过3种方式添加全文模糊搜索-程序员宅基地

文章浏览阅读561次。背景介绍 最近在给博客做SEO,顺便也学学怎么与搜索引擎打交道 发现自己站内没有一个像样的搜索功能 于是全文搜索便是第一步给博客文章添加全文搜索,摸索着尝试了3种方案 可以根据具体项目选择 数据库:Mysql 1.SQL Like 查询 ~初级方案~使用作为最简单的方式,直接使用like条件在ttitle和content中查询 select 优势1.简单 2.简单 3.简单劣势1.效率低下,查询..._博客搜索功能实现

hive on spark hql 插入数据报错 Failed to create Spark client for Spark session Error code 30041_rg.apache.hadoop.hive.ql.exec.spark.sparktask. fai-程序员宅基地

文章浏览阅读6.1k次,点赞13次,收藏25次。Failed to execute spark task, with exception 'org.apache.hadoop.hive.ql.metadata.HiveException(Failed to create Spark client for Spark session 50cec71c-2636-4d99-8de2-a580ae3f1c58)'FAILED: Execution Error, return code 30041 from org.apache.hadoop.hive.ql._rg.apache.hadoop.hive.ql.exec.spark.sparktask. failed to create spark client

官方安装文档解读SAP S4 HANA架构_s4 hana architecture-程序员宅基地

文章浏览阅读2.5k次。官方安装文档解读SAP S4 HANA架构SAP HANA原生备份与恢复自己攒一台SAP S4 HANA服务器硬件配置参考BASIS模块管理为什么SAP ECC的系统,数据库总是和SAP的应用安装在一起SAP PI服务器,版本731 SYBASE数据库运维故障解决2粒ERP-SAP服务器集群架构技术沿革IBM小型机厉害的高级功能AIX下 SYBASE 数据库无法启动..._s4 hana architecture

VTK_9.0.0 空间两点间的点云距离距离计算_vtkdistancewidget-程序员宅基地

文章浏览阅读1.2k次。长度测量其中新建项目配置好VTK所需的环境,运行程序报错,Error: no override found for 'vtkPolyDataMapper2D'.This initializes the specified VTK modules. CMake includes these by default, but other compilers such as VS do not...._vtkdistancewidget

「sumo carla自动驾驶联合仿真安装配置教程:开发驾驶模拟、强化学习、轨迹预测和规划」-程序员宅基地

文章浏览阅读267次,点赞5次,收藏4次。摘要:本文将介绍Sumo和Carla两个开源仿真平台的安装、配置和使用教程,并结合强化学习的方法,探讨在自动驾驶场景下的轨迹预测和规划技术。通过对Sumo和Carla联合仿真的实践与分析,我们将展示如何利用这两个平台的优势,为自动驾驶系统的开发与研究提供一种高效、可靠的模拟环境。通过对自动驾驶场景下的强化学习轨迹预测和规划的实践,我们将展示Sumo和Carla的联合仿真能够为自动驾驶系统的开发与研究提供高效、可靠的模拟环境。关键词:Sumo,Carla,自动驾驶,联合仿真,强化学习,轨迹预测,轨迹规划。

随便推点

iOS开发 --- 添加一个全局悬浮按钮_ios开发 悬浮-程序员宅基地

文章浏览阅读6.8k次。背景介绍 :在普通的iOS开发组中,一般测试机都不止一台,但是我们在开发的时候,不可能每台测试机时刻保持最新的代码,这就出现了一个问题,当测试测出问题的时候,(或者产品突然拿去点点看的时候出了问题)如果不知道当前的版本,可能不确定是什么时候出的问题。解决方案:如果当前环境是测试服的时候,展示一个全局浮动标签,这样不仅看到此标志就告诉测试(包括我们自己)当前的环境,当出现问题的时候,通过标签,可以快..._ios开发 悬浮

eclipse可以start模式可以启动项目但debug模式不能启动_eclipse tomcat debug就报错-程序员宅基地

文章浏览阅读577次。去掉所有断点即可解决问题。原因可能eclipse和tomcat的交互而产生的,在以debug模式启动tomcat时,发生了读取文件错误,eclipse自动设置了断点,导致tomcat不能正常启动。解决方法如下:以debug模式启动tomcat,打开breakpoints veiw,右键-> Remove all,重启下tomcat就OK了。_eclipse tomcat debug就报错

FreeMarker教程-程序员宅基地

文章浏览阅读330次。一、什么是模板引擎,为什么要用模板引擎在B/S程式设计中,常常有美工和程序员二个角色,他们具有不同专业技能:美工专注于表现——创建页面、风格、布局、效果等等可视元素;而程序员则忙于创建程式的商业流程,生成设计页面要显示的数据等等。很多时候,要显示的资料在设计的时候并不存在,它们一般是在运行时由程式产生的,比如执行“价格不高于800NT的USB Disk”查询的返回结果。这种技术需求产生了J..._freemarker教程

OpenCV+CUDA学习2---图像灰度化_opencv gpu灰度化-程序员宅基地

文章浏览阅读2.9k次。对于图像灰度化,使用了opencv-cuda实现与完全基于cuda实现,本程序中参考了网上多个教程,主要记录和学习的过程。目录1、opencv+cuda实现图像灰度化2、CUDA实现图像灰度化3、图像展示3.1 opencv+cuda3.2 cuda1、opencv+cuda实现图像灰度化#include <iostream>#include <opencv2/opencv.hpp>#include <opencv2/cudaarith_opencv gpu灰度化

网页导出的excel无法计算机,网页上不能导出excel表格数据-如何将网页表格导出到excel...-程序员宅基地

文章浏览阅读7.9k次。如何将网页表格导出到excel1、打开电脑,进入任意一个浏览器即可,首先找到浏览器【工具】栏菜单,打开浏览器【Internet选项】;2、点击【安全】选项卡,之后会出现安全设置界面;3、点击下方的自定义级别,在弹出的窗口中找到”对未标记为可安全执行的脚本的ActiveX控件初始化并执行脚本“这一选项;4、点击启用;之后将弹出警告窗口,选择“是”,随后将自动返回一级菜单,工具栏下方将显示“您的安全设..._不安全网站导出怎么弄

ArcGIS Server 10.8.1安装-程序员宅基地

文章浏览阅读1w次,点赞4次,收藏31次。官方安装文档:ArcGIS Server 系统要求—ArcGIS Enterprise | ArcGIS Enterprise 文档_arcgis server 10.8

推荐文章

热门文章

相关标签