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

技术标签: 回调函数  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

智能推荐

【数据结构】查找1——线性表的查找(顺序查找、折半查找、分块查找)_图顺序查找-程序员宅基地

文章浏览阅读2.8k次。【数据结构】查找1——线性表的查找(顺序查找、折半查找、分块查找)_图顺序查找

zip包解压时报malformed input off : 0, length : 1-程序员宅基地

文章浏览阅读843次。使用ZipArchiveInputStream而非ZipInputStream的原因主要有以下两点:支持更多的压缩格式:ZipArchiveInputStream是Apache Commons Compress库中提供的类,能够支持多种压缩格式,包括Zip、Gzip、Tar、Jar等。而ZipInputStream是Java标准库中的类,只能读取普通的Zip文件。因此,如果需要处理多种压缩格式的文件,使用ZipArchiveInputStream会更加方便。更多的选项和功能:ZipArchiveInp_malformed input off : 0, length : 1

switch case结合枚举值使用,借助枚举的值来做case分支判断_switch case 枚举-程序员宅基地

文章浏览阅读8.3k次,点赞9次,收藏22次。https://blog.csdn.net/m0_37754981/article/details/80022169参考资料_switch case 枚举

rpc简介及原理-程序员宅基地

文章浏览阅读5.1k次,点赞2次,收藏30次。1.RPC简介及原理介绍RPC技术内部原理是通过两种技术的组合来实现的:本地方法调用 和 网络通信技术。1.1 RPC简介在上述本地过程调用的例子中,我们是在一台计算机上执行了计算机上的程序,完成调用。随着计算机技术的发展和需求场景的变化,有时就需要从一台计算机上执行另外一台计算机上的程序的需求,因此后来又发展出来了RPC技术。特别是目前随着互联网技术的快速迭代和发展,用户和需求几乎都是以指数式的方式在高速增长,这个时候绝大多数情况下程序都是部署在多台机器上,就需要在调用其他物理机器上的程序的情况。_rpc

AI大视觉(十七) | PANet(路径聚合网络)-程序员宅基地

文章浏览阅读7k次,点赞6次,收藏70次。本文来自公众号“AI大道理​目标检测或者实例分割不仅要关心语义信息,还要关注图像的精确到像素点的浅层信息。所以需要对骨干网络中的网络层进行融合,使其同时具有深层的语义信息和浅层的纹理信息。PANet整体结构PANet(Path Aggregation Network)最大的贡献是提出了一个自顶向下和自底向上的双向融合骨干网络,同时在最底层和最高层之间添加了一条“short-cut”,用于缩短层之间的路径。PANet还提出了自适应特征池化和全连接融合两个模块。其中自适应特征_panet

OperationalError: (sqlite3.OperationalError) unable to open database file解决方案_sqlite3.operationalerror: unable to open database -程序员宅基地

文章浏览阅读1.3w次,点赞13次,收藏14次。本文主要介绍了OperationalError: (sqlite3.OperationalError) unable to open database file解决方案,希望能对使用langchain读取sqlite文件的同学们有所帮助。文章目录1. 问题描述2. 解决方案_sqlite3.operationalerror: unable to open database file

随便推点

PID控制详解-程序员宅基地

文章浏览阅读10w+次,点赞551次,收藏3.2k次。PID控制详解一、PID控制简介 PID( Proportional Integral Derivative)控制是最早发展起来的控制策略之一,由于其算法简单、鲁棒性好和可靠性高,被广泛应用于工业过程控制,尤其适用于可建立精确数学模型的确定性控制系统。 在工程实际中,应用最为广泛的调节器控制规律为比例、积分、微分控制,简称PID控制,又称PID调节,它实际上是一种算法。PID控制器问..._pid控制

搭建静态网站-程序员宅基地

文章浏览阅读329次。搭建Http静态服务器环境任务时间:15min ~ 30min搭建静态网站,首先需要部署环境。下面的步骤,将告诉大家如何在服务器上通过 Nginx 部署 HTTP 静态服务。00、安装 Nginx在 CentOS 上,可直接使用yum来安装 Nginxyum install nginx -y安装完成后,使用nginx命令启动 Nginx:n..._在一个目录下创建静态站点

HTML5网页设计 (一)_html5网页制作-程序员宅基地

文章浏览阅读7k次,点赞2次,收藏23次。初始HTML5记事本简单实现HTML5页面_html5网页制作

CentOS 固定IP配置_centos配置ip地址-程序员宅基地

文章浏览阅读6.8k次。1、打开 VMware Workstation,点击 “编辑” -> "虚拟网络编辑器"2、选择 “更改设置”(若界面内没有,则不用理会)3、选择 “VMnet8”,更改为192.168.88.0(按需填写)、为255.255.255.0,点击 “NAT设置”4、更改为192.168.88.2(按需填写),当前界面点击确定,“虚拟网络编辑器”界面再点击确定。_centos配置ip地址

MATLAB 调用 p文件,Matlab中使用.p文件的方法-程序员宅基地

文章浏览阅读2.9k次。首先,P文件为了保护知识产权设计的一种加密文件,是不能查看的。运行比较简单,和调用m文件方法一样。下面是一些解释。P文件是对应M文件的一种预解析版本(preparsedversion)。因为当你第一次执行M文件时,Matlab需要将其解析(parse)一次(第一次执行后的已解析内容会放入内存作第二次执行时使用,即第二次执行时无需再解析),这无形中增加了执行时间。所以我们就预先作解释,那么以后再使用..._matlab怎么调用p文件

[CSS] 内联元素内的文字居中显示(按钮、链接等)_a-button居中-程序员宅基地

文章浏览阅读2.1k次。最近在做一个移动端的响应式页面,对自定义样式要求较高,很多地方需要细微的调整,比如按钮、链接等内联元素,不能只是让它看起来居中,而要做到“响应式”居中……水平居中很简单,但是垂直居中的问题困扰了我很久,直到我研究过bootstrap的_variables.scss文档后,才仔细地了解到一些组件的构造。如果修改了inline/inline-block元素的默认样式后,里面的文字不居中了,试试这个方法,只需要5步~_a-button居中

推荐文章

热门文章

相关标签