0

0

C++如何实现回调函数?(函数指针与std::function)

冰火之心

冰火之心

发布时间:2026-01-21 14:59:02

|

834人浏览过

|

来源于php中文网

原创

C++回调函数核心是传递可调用逻辑,常用函数指针(轻量无状态)和std::function(灵活支持闭包与成员函数);函数指针适用于固定签名、无捕获的C风格API或性能敏感场景。

c++如何实现回调函数?(函数指针与std::function)

在C++中实现回调函数,核心是把“一段可调用的逻辑”作为参数传给另一个函数,让后者在合适时机执行它。常用方式有两种:传统函数指针(轻量、无状态)和 std::function(灵活、支持闭包与成员函数)。

用函数指针实现简单回调

函数指针适合固定签名、无捕获需求的场景,比如 C 风格 API 的兼容或性能敏感路径。

定义方式是声明一个指向特定参数/返回类型的函数的指针类型:

void on_click(int x, int y) {
    std::cout << "Clicked at (" << x << ", " << y << ")\n";
}

// 类型别名,提高可读性 using ClickHandler = void(*)(int, int);

void register_click_handler(ClickHandler handler) { // 保存或立即调用 if (handler) handler(100, 200); }

调用时直接传函数名(自动退化为指针):

立即学习C++免费学习笔记(深入)”;

register_click_handler(on_click);

⚠️ 注意:函数指针不能绑定局部变量、lambda(除非无捕获且显式转函数指针)、成员函数。

LobeHub
LobeHub

LobeChat brings you the best user experience of ChatGPT, OLLaMA, Gemini, Claude

下载

用 std::function 实现通用回调

std::function 是类型擦除容器,能容纳任意可调用对象(普通函数、lambda、bind 表达式、成员函数指针等),只要签名匹配。

使用前需包含头文件:#include

典型用法:

#include 
#include 

void run_callback(std::function cb) { if (cb) cb(42); }

int main() { // 普通函数 run_callback([](int x) { std::cout << "Lambda: " << x << "\n"; });

// 带捕获的 lambda(函数指针做不到)
int offset = 10;
run_callback([offset](int x) { std::cout << "With offset: " << x + offset << "\n"; });

// 绑定成员函数
struct Button {
    void click(int code) { std::cout << "Button code: " << code << "\n"; }
};
Button btn;
run_callback(std::bind(&Button::click, &btn, std::placeholders::_1));

}

处理成员函数回调

类成员函数有隐式 this 参数,不能直接当函数指针用。常见解法有三种:

  • 用 std::function + lambda 捕获 this:最直观,推荐用于类内回调注册
  • 用 std::bind 绑定对象和成员函数:明确表达绑定关系,但语法稍冗长
  • 静态成员函数 + void* 用户数据:C 风格兼容方案,需手动管理生命周期,易出错

例如,在类中注册自身方法为回调:

class EventManager {
public:
    void set_handler(std::function h) { handler_ = std::move(h); }
    void trigger() { if (handler_) handler(); }

private: std::function handler_; };

struct MyWidget { void handle_event() { std::cout << "Handled!\n"; }

void setup() {
    EventManager mgr;
    // 捕获 this,安全调用成员函数
    mgr.set_handler([this]() { this->handle_event(); });
    mgr.trigger();
}

};

选择建议与注意事项

  • 只调用全局/静态函数,且追求零开销 → 选函数指针
  • 需要 lambda、成员函数、状态捕获或统一接口 → 必选 std::function
  • std::function 有轻微运行时开销(类型擦除、可能堆分配),但现代编译器对简单 lambda 常做优化
  • 避免存储临时 lambda 的 std::function 并长期持有——若 lambda 捕获了局部变量,对象销毁后调用会崩溃
  • 考虑用模板参数替代 std::function 实现编译期多态(如接受任意 Callable),进一步避免运行时成本

相关专题

更多
java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

15

2025.11.27

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

97

2025.11.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

205

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

49

2026.01.05

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1047

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

86

2025.10.17

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

9

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Django 教程
Django 教程

共28课时 | 3.3万人学习

Excel 教程
Excel 教程

共162课时 | 12.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号