当前位置: 首页 > news >正文

浅探C语言的回调函数(Callback Function)

目录

前言

解析回调函数组成

        1.没有参数传递的回调函数

        2.有参数传递的回调函数 

使用

结语

 


前言

C语言里面有回调函数的用法,只要将函数的指针传入,对应的函数即可运行。这里我也贴一下其它解释:

回调函数本质是 "函数指针的参数化",通过将行为抽象为函数指针参数,实现动态控制流程。这种机制是C语言实现事件驱动、异步操作和通用算法的基石,广泛应用于系统编程、库设计和嵌入式开发中。


 

典型使用场景:

  1. 事件处理(如GUI库中的按钮点击)

  2. 异步操作(如I/O操作完成后的通知)

  3. 算法通用化(如排序函数qsort()中的自定义比较逻辑)

  4. 库函数设计(允许用户注入自定义行为)

本文我们研究研究这个回调函数。

解析回调函数组成

这里我提供两个比较通用的回调函数,一个是没有参数的回调,一个是有参数的回调。

//无参数的回调函数
void callback(void *fun)
{((void *(*)())(fun))();
}//有参数的回调函数
void callback(void *fun, void *arg)
{((void *(*)(void *))(fun))(arg);
}

我们主要研究函数里面的那两个表达式,这才是回调函数的精髓。 

1.没有参数传递的回调函数

((void *(*)())(fun))();

这里其实有一个强制转换的过程

(void *(*)())

没错,括号里面其实是一个强制转换,转换为返回值为void *类型,(*)表明这里对应位置是一个指针,()表示没有参数。

然后在函数中对应

(fun)();

 最后就是对参数的处理,对转换完成的函数指针,表明没有参数。

((void *(*)())(fun))---->();

2.有参数传递的回调函数 

((void *(*)(void *))(fun))(arg);

同理,这里也有一个强制转换的过程

(void *(*)(void *))

强制转换为返回值为void *,(*)对应位置还是函数的指针,(void *)代表接收的参数是void *类型。

然后就是将转换之后的fun说明是那个参数arg 

(fun)(arg)

先转换,转换之后再赋上参数的值。

((void *(*)(void *))(fun))(arg);

使用

这里我贴的虽然是C++的代码,其实只是函数重载比较方便,改个函数名,一样能使用!

#include <iostream>
#include <unistd.h>
void callback(void *fun);
void callback(void *fun, void *arg);void *task(void *arg)
{long int num = (long int)arg;while (1){std::cout << "Hello!World!"<<"-->"<<num << std::endl;sleep(1);num++;}
}int main(int argc, char const *argv[])
{long int i = 0;callback((void *)task,(void *)i);return 0;
}void callback(void *fun)
{((void *(*)())(fun))();
}void callback(void *fun, void *arg)
{((void *(*)(void *))(fun))(arg);
}

结语

在C语言中,回调函数(Callback Function) 是一个通过函数指针调用的函数。其核心思想是将一个函数的地址作为参数传递给另一个函数(通常称为"调用函数"或"高阶函数"),当特定事件或条件发生时,调用函数会通过这个指针调用回调函数。

核心概念:

  1. 函数指针传递:回调函数的地址被作为参数传递给另一个函数。

  2. 异步执行:回调函数不会立即执行,而是在特定条件(如事件触发、任务完成)时被调用。

  3. 解耦与扩展性:调用函数无需知道回调函数的具体实现,只需定义接口,使代码更灵活。

http://www.dtcms.com/a/284199.html

相关文章:

  • CD54.【C++ Dev】vector和list的反向迭代器的实现
  • 大语言模型Gemini Deep Research 全流程解读+使用攻略
  • uniapp云打包安卓
  • ADVB时序图
  • Spring Boot基于AOP的本地/远程调用动态路由实践
  • 如何在银河麒麟桌面系统中启用 sudo 密码的星号反馈
  • QT之openGL使用(一)
  • 燕之屋打造多元化产品组合,引领行业迈向高质量发展新里程
  • 心通达OA知识管理平台:高效解决单位知识管理难题
  • 点晴模切ERP帮忙模切行业向数智化管理转型
  • Rocky Linux 9 源码包安装php7
  • 如何通过mac的前24bit,模糊确认是那一台什么样的设备
  • macOS 字体管理全攻略:如何查看已安装字体及常见字体格式区
  • 从基础到实战:.NET 反射机制的进阶用法与最佳实践
  • Tekla多部门协作,授权资源如何共享与调度?
  • 暑期算法训练.3
  • day29:零基础学嵌入式之线程1.0
  • HTML 极简个人介绍卡片(侧重语义化标签和响应式布局)
  • pytorch小记(三十二):深度解析 PyTorch 的 `torch.remainder`:向下取整余数运算
  • 【web安全】DVWA存储型XSS分析与利用
  • 第6天| openGauss中用户一次只能连接到一个数据库,没法访问其他数据库的对象
  • arping(ARP协议网络测试工具)
  • 【实时Linux实战系列】实时系统的安全性架构
  • MySQL如何解决事务并发的幻读问题
  • 从单线程到云原生:Redis 二十年演进全景与内在机理深剖
  • RuoYi-Cloud 定制微服务
  • 宝塔申请证书错误,提示 module ‘OpenSSL.crypto‘ has no attribute ‘sign‘
  • 有痛呻吟!!!
  • 09-three.js Materials
  • 任务4.1 谁做的好事