C++ 【回调函数】详解与代码解读

news2025/1/5 20:30:34

在现代软件开发中,回调函数是一个常用的工具,能够实现函数调用的延迟绑定,广泛应用于事件驱动、异步操作以及模块解耦等场景。本文将从基础概念、分类、实现方式到代码示例,全面讲解 C++ 回调函数的实现和应用。


什么是回调函数?

回调函数(Callback Function)是指一个函数通过函数指针或其他机制传递给另一个函数,在执行过程中由该函数调用。这种调用方式赋予了程序更大的灵活性,因为调用方无需知道具体的函数实现,只需在需要时触发回调。

一个生活中的例子

假设你使用外卖平台(一个函数)下单,平台会在订单完成后给你打电话(回调函数)。这个电话的内容可能是你指定的(动态绑定),但外卖平台不需要了解具体的细节,只负责在指定时间触发。


C++ 回调函数的分类

根据实现方式和场景,C++ 回调函数可以分为以下几类:

  1. 普通函数的回调

    • 通过函数指针实现,是最基础的回调方式。
  2. 类成员函数的回调

    • 普通成员函数:需要对象指针和成员函数指针配合使用。
    • 静态成员函数:可作为普通函数指针使用,简单灵活。
  3. 基于 std::functionstd::bind 的回调

    • 使用 C++11 引入的标准库实现,能够兼容普通函数、成员函数和 Lambda 表达式。
  4. Lambda 表达式的回调

    • C++11 引入的匿名函数机制,适用于简洁的回调逻辑。

实现回调函数的几种方式

1. 普通函数的回调

普通函数的回调是通过函数指针来实现的。函数指针保存了一个函数的地址,调用时可以直接跳转到该函数。

#include <iostream>
using namespace std;

// 普通函数,作为回调函数
void callbackFunction(int value) {
    // 回调逻辑
    cout << "普通函数回调,被调用时传入的值是: " << value << endl;
}

// 一个执行回调的函数
void executeCallback(void (*callback)(int), int value) {
    // 通过函数指针调用回调函数
    callback(value);
}

int main() {
    // 将普通函数的地址(指针)传入
    executeCallback(callbackFunction, 42);
    return 0;
}
代码解析:
  1. callbackFunction 是普通函数,定义了回调的逻辑。
  2. executeCallback 是一个高阶函数,它接收函数指针 void (*callback)(int) 作为参数,并在内部调用回调函数。
  3. main 中,将函数 callbackFunction 的地址传递给 executeCallback,实现回调。
输出:
普通函数回调,被调用时传入的值是: 42

适用场景:适合简单的场景,例如小型工具程序。


2. 类成员函数的回调

由于普通成员函数必须依赖具体对象才能调用,因此在回调时需要传递对象指针和成员函数指针。

#include <iostream>
using namespace std;

class CallbackHandler {
public:
    // 成员函数作为回调函数
    void memberCallback(int value) {
        cout << "成员函数回调,被调用时传入的值是: " << value << endl;
    }
};

// 一个执行回调的函数
void executeMemberCallback(CallbackHandler* obj, void (CallbackHandler::*callback)(int), int value) {
    // 通过对象指针和成员函数指针调用成员函数
    (obj->*callback)(value);
}

int main() {
    CallbackHandler handler; // 创建一个对象
    // 传递对象和成员函数指针进行回调
    executeMemberCallback(&handler, &CallbackHandler::memberCallback, 42);
    return 0;
}
代码解析:
  1. CallbackHandler 类中定义了一个成员函数 memberCallback
  2. executeMemberCallback 接收对象指针 CallbackHandler* obj 和成员函数指针 void (CallbackHandler::*callback)(int),通过 (obj->*callback)(value) 调用成员函数。
  3. main 函数中,通过对象 handler 和成员函数指针 &CallbackHandler::memberCallback 实现回调。
输出:
成员函数回调,被调用时传入的值是: 42

适用场景:适用于面向对象的程序,尤其是需要调用类成员函数的情况。


3. 基于 std::functionstd::bind 的回调

std::function 是现代 C++ 中推荐的工具,可以存储任意可调用对象(普通函数、成员函数、Lambda 表达式等),而 std::bind 可以绑定成员函数与对象。

#include <iostream>
#include <functional> // 包含 std::function 和 std::bind
using namespace std;

// 一个执行回调的函数
void executeCallback(std::function<void(int)> callback, int value) {
    // 调用回调
    callback(value);
}

void freeFunction(int value) {
    cout << "普通函数回调,被调用时传入的值是: " << value << endl;
}

class CallbackHandler {
public:
    void memberCallback(int value) {
        cout << "成员函数回调,被调用时传入的值是: " << value << endl;
    }
};

int main() {
    CallbackHandler handler;

    // 1. 传递普通函数作为回调
    executeCallback(freeFunction, 42);

    // 2. 传递 Lambda 表达式作为回调
    executeCallback([](int value) {
        cout << "Lambda 回调,被调用时传入的值是: " << value << endl;
    }, 43);

    // 3. 传递成员函数作为回调
    executeCallback(std::bind(&CallbackHandler::memberCallback, &handler, std::placeholders::_1), 44);

    return 0;
}
代码解析:
  1. std::function<void(int)> 可以存储普通函数、Lambda 表达式或绑定后的成员函数。
  2. 使用 std::bind 将成员函数和对象绑定,并通过 std::placeholders::_1 占位符传递参数。
输出:
普通函数回调,被调用时传入的值是: 42
Lambda 回调,被调用时传入的值是: 43
成员函数回调,被调用时传入的值是: 44

适用场景:适合复杂场景,尤其是需要兼容不同类型回调的情况。


4. Lambda 表达式的回调

Lambda 表达式是 C++11 引入的一种简洁的回调实现方式,适合定义小型的、一次性的回调逻辑。

#include <iostream>
using namespace std;

// 一个执行回调的函数
void executeCallback(auto callback, int value) {
    callback(value);
}

int main() {
    // 使用 Lambda 表达式定义回调
    executeCallback([](int value) {
        cout << "Lambda 表达式回调,被调用时传入的值是: " << value << endl;
    }, 42);

    return 0;
}
代码解析:
  1. Lambda 表达式是一种匿名函数,可以直接在需要的地方定义。
  2. 使用 auto 参数,避免显式定义函数类型。
输出:
Lambda 表达式回调,被调用时传入的值是: 42

适用场景:适用于简单的回调逻辑,代码更加简洁。


总结

实现方式优点缺点适用场景
普通函数的回调简单直观灵活性不足小型工具程序
类成员函数的回调面向对象,支持类的封装调用复杂,需要传递对象和函数指针面向对象程序
std::functionstd::bind强大灵活,支持多种可调用对象性能略低于直接函数指针复杂回调场景
Lambda 表达式的回调简洁直观,代码更加紧凑不适合复杂逻辑小型回调逻辑

在实际开发中,建议优先使用 Lambda 表达式std::function,它们在现代 C++ 编程中更易读、易维护,同时具有更好的兼容性。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

No.1十六届蓝桥杯备战|第一个C++程序|cin和cout|命名空间

第一个C程序 基础程序 使用DevC5.4.0 写一个C程序 在屏幕上打印hello world #include <iostream> using namespace std;int main() {cout << "hello world" << endl;return 0; } 运行这个C程序 F9->编译 F10->运行 F11->编译运行 mai…

【Vim Masterclass 笔记05】第 4 章:Vim 的帮助系统与同步练习

文章目录 Section 4&#xff1a;The Vim Help System&#xff08;Vim 帮助系统&#xff09;S04L14 Getting Help1 打开帮助系统2 退出帮助系统3 查看具体命令的帮助文档4 查看帮助文档中的主题5 帮助文档间的上翻、下翻6 关于 linewise7 查看光标所在术语名词的帮助文档8 关于退…

印象笔记07——试一试PDF标注

印象笔记07——试一试PDF标注 [!CAUTION] 根据第六期&#xff0c;我再次查询了资料&#xff0c;印象笔记还是有一些可圈可点的功能的&#xff08;当然部分有平替&#xff09;&#xff0c;针对会员作用&#xff0c;开发使用场景虽然是逆向的&#xff0c;但我坚信这是一部分人的现…

JDK17源码分析Jdk动态代理底层原理

本文侧重分析JDK17中jdk动态代理的源码&#xff0c;若是想看JDK8源码分析可以看我的这一篇文章 JDK8源码分析Jdk动态代理底层原理-CSDN博客 两者之间有着略微的差别&#xff0c;JDK17在JDK8上改进了不少 目录 JDK 17的动态代理源码 核心入口方法 newProxyInstance 获取代理类…

【网络协议】开放式最短路径优先协议OSPF详解(一)

OSPF 是为取代 RIP 而开发的一种无类别的链路状态路由协议&#xff0c;它通过使用区域划分以实现更好的可扩展性。 文章目录 链路状态路由协议OSPF 的工作原理OSPF 数据包类型Dijkstra算法、管理距离与度量值OSPF的管理距离OSPF的度量值 链路状态路由协议的优势拓扑结构路由器O…

vim 的基础使用

目录 一&#xff1a;vim 介绍二&#xff1a;vim 特点三&#xff1a;vim 配置四&#xff1a;vim 使用1、vim 语法格式2、vim 普通模式&#xff08;1&#xff09;保存退出&#xff08;2&#xff09;光标跳转&#xff08;3&#xff09;文本删除&#xff08;4&#xff09;文本查找&…

为什么深度学习和神经网络要使用 GPU?

为什么深度学习和神经网络要使用 GPU&#xff1f; 本篇文章的目标是帮助初学者了解 CUDA 是什么&#xff0c;以及它如何与 PyTorch 配合使用&#xff0c;更重要的是&#xff0c;我们为何在神经网络编程中使用 GPU。 图形处理单元 (GPU) 要了解 CUDA&#xff0c;我们需要对图…

工厂模式与抽象工厂模式在Unity中的实际应用案例

一、实验目的 实践工厂模式和抽象工厂模式的实际应用。 创建一个小型的游戏场景&#xff0c;通过应用这些设计模式提升游戏的趣味性和可扩展性。 掌握在复杂场景中管理和使用不同类型的对象。 比较在实际游戏开发中不同设计模式的实际效果和应用场景。 学习如何进行简单的性…

jrc水体分类对水体二值掩码修正

使用deepwatermap生成的水体二值掩码中有部分区域由于被云挡住无法识别&#xff0c;造成水体不连续是使用jrc离线数据进行修正&#xff0c;jrc数据下载连接如下&#xff1a;https://global-surface-water.appspot.com/download 选择指定区域的数据集合下载如图&#xff1a; 使…

计算机网络 (20)高速以太网

一、发展背景 随着计算机技术和网络应用的不断发展&#xff0c;传统的以太网速率已逐渐无法满足日益增长的带宽需求。因此&#xff0c;高速以太网应运而生&#xff0c;它以提高数据传输速率为主要目标&#xff0c;不断推动着以太网技术的发展。 二、技术特点 高速传输&#xff…

基于SpringBoot的校园二手交易平台的设计与实现(源码+SQL+LW+部署讲解)

文章目录 摘 要1. 第1章 选题背景及研究意义1.1 选题背景1.2 研究意义1.3 论文结构安排 2. 第2章 相关开发技术2.1 前端技术2.2 后端技术2.3 数据库技术 3. 第3章 可行性及需求分析3.1 可行性分析3.2 系统需求分析 4. 第4章 系统概要设计4.1 系统功能模块设计4.2 数据库设计 5.…

2024年中国新能源汽车用车发展怎么样 PaperGPT(二)

用车趋势深入分析 接上文&#xff0c;2024年中国新能源汽车用车发展怎么样 PaperGPT&#xff08;一&#xff09;-CSDN博客本文将继续深入探讨新能源汽车的用车强度、充电行为以及充电设施的现状。 用车强度 月均行驶里程&#xff1a;2024年纯电车辆月均行驶超过1500公里&…

antd-vue - - - - - a-date-picker限制选择范围

antd-vue - - - - - a-date-picker限制选择范围 1. 效果展示2. 代码展示 1. 效果展示 如图&#xff1a;限制选择范围为 今年 & 去年 的 月份. 2. 代码展示 <template><a-date-picker:disabledDate"disabledDate"picker"month"/> &l…

滑动窗口、流量控制和拥塞控制

1. 确认应答机制 确认应答机制是计算机网络中&#xff0c;用于确保数据可靠传输的一种方法。 它通过发送 ACK 数据段来通知对方&#xff0c;每一个 ACK 数据段都有一个确认序号&#xff0c;表明&#xff1a; 确认序号之前的所有数据都已被接收&#xff0c;接下来从确认序号开…

TCP粘/拆包----自定义消息协议

今天是2024年12月31日&#xff0c;今年的最后一天&#xff0c;希望所有的努力在新的一年会有回报。❀ 无路可退&#xff0c;放弃很难&#xff0c;坚持很酷 TCP传输 是一种面向二进制的&#xff0c;流的传输。在传输过程中最大的问题是消息之间的边界不明确。而在服务端主要的…

前端,npm install安装依赖卡在sill idealTree buildDeps(设置淘宝依赖)

输入npm i后&#xff0c;一直卡在sill idealTree buildDeps&#xff0c;一动不动 cnpm可以安装成功&#xff0c;但使用cnpm不会生成package-lock.json文件 设置淘宝依赖&#xff0c;依然卡住&#xff0c;挂梯子也不行 解决方法&#xff1a; // 取消ssl验证 set strict-ssl …

【有作图代码】Highway Network与ResNet:skip connection如何解决深层网络欠拟合问题

【有作图代码】Highway Network与ResNet&#xff1a;skip connection如何解决深层网络欠拟合问题 关键词&#xff1a; #Highway Network #ResNet #skip connection #深层网络 #欠拟合问题 具体实例与推演 假设我们有一个深层神经网络&#xff0c;其层数为L&#xff0c;每一…

目标检测入门指南:从原理到实践

目录 1. 数据准备与预处理 2. 模型架构设计 2.1 特征提取网络原理 2.2 区域提议网络(RPN)原理 2.3 特征金字塔网络(FPN)原理 2.4 边界框回归原理 2.5 非极大值抑制(NMS)原理 2.6 多尺度训练与测试原理 2.7 损失函数设计原理 3. 损失函数设计 4. 训练策略优化 5. 后…

搭建开源版Ceph分布式存储

系统&#xff1a;Rocky8.6 三台2H4G 三块10G的硬盘的虚拟机 node1 192.168.2.101 node2 192.168.2.102 node3 192.168.2.103 三台虚拟机环境准备 1、配置主机名和IP的映射关系 2、关闭selinux和firewalld防火墙 3、配置时间同步且所有节点chronyd服务开机自启 1、配置主机名和…

租用服务器还是服务器托管:哪种方案更适合您?

随着企业对网络服务质量要求的不断提高&#xff0c;租用服务器和服务器托管是两种常见的选择&#xff0c;各自具备独特的优势和适用场景。这篇文章将从多个维度对这两种方案进行详细分析&#xff0c;帮助大家进行对比选择。 租用服务器的优劣势分析 优点 无需大额初始投入 租用…