回调函数的概念

回调函数的基本概念

回调函数的本质是"你定义,我调用" - 你提供一个函数,系统或框架在特定时机调用它。

实现回调函数的几种方式

1. 函数指针(传统方式)

复制代码
#include <iostream>

// 回调函数类型定义
typedef void (*Callback)(int, const std::string&);

// 接受回调函数的函数
void processData(int value, const std::string& name, Callback callback) {
    std::cout << "处理数据中..." << std::endl;
    // 处理完成后调用回调函数
    callback(value, name);
}

// 具体的回调函数实现
void myCallback(int num, const std::string& text) {
    std::cout << "回调函数被调用: " << num << ", " << text << std::endl;
}

int main() {
    processData(42, "Hello", myCallback);
    return 0;
}

2. std::function(现代方式)

复制代码
#include <iostream>
#include <functional>

void processData(int value, const std::string& name, 
                std::function<void(int, const std::string&)> callback) {
    std::cout << "处理数据中..." << std::endl;
    callback(value, name);
}

void myCallback(int num, const std::string& text) {
    std::cout << "回调函数被调用: " << num << ", " << text << std::endl;
}

int main() {
    // 使用函数指针
    processData(42, "World", myCallback);
    
    // 使用lambda表达式
    processData(100, "Lambda", [](int n, const std::string& s) {
        std::cout << "Lambda回调: " << n << ", " << s << std::endl;
    });
    
    return 0;
}

3. 函数对象(Functor)

复制代码
#include <iostream>

class MyCallback {
public:
    void operator()(int value, const std::string& message) {
        std::cout << "函数对象回调: " << value << ", " << message << std::endl;
    }
};

template<typename CallbackType>
void processData(int value, const std::string& name, CallbackType callback) {
    std::cout << "处理数据中..." << std::endl;
    callback(value, name);
}

int main() {
    MyCallback callbackObj;
    processData(200, "Functor", callbackObj);
    return 0;
}

实际应用示例

事件处理系统

复制代码
#include <iostream>
#include <functional>
#include <vector>

class Button {
private:
    std::vector<std::function<void()>> clickCallbacks;

public:
    // 注册点击回调
    void onClick(std::function<void()> callback) {
        clickCallbacks.push_back(callback);
    }
    
    // 模拟按钮点击
    void click() {
        std::cout << "按钮被点击!" << std::endl;
        for (auto& callback : clickCallbacks) {
            callback();
        }
    }
};

int main() {
    Button btn;
    
    // 注册多个回调函数
    btn.onClick([]() {
        std::cout << "执行操作1: 保存文件" << std::endl;
    });
    
    btn.onClick([]() {
        std::cout << "执行操作2: 更新界面" << std::endl;
    });
    
    // 模拟点击
    btn.click();
    
    return 0;
}

异步操作回调

复制代码
#include <iostream>
#include <functional>
#include <thread>
#include <chrono>

void asyncOperation(std::function<void(bool, const std::string&)> callback) {
    std::thread([callback]() {
        std::cout << "开始异步操作..." << std::endl;
        std::this_thread::sleep_for(std::chrono::seconds(2));
        
        // 模拟操作完成
        bool success = true; // 或 false
        std::string result = "操作完成";
        
        callback(success, result);
    }).detach();
}

int main() {
    std::cout << "主线程继续执行..." << std::endl;
    
    asyncOperation([](bool success, const std::string& message) {
        if (success) {
            std::cout << "成功: " << message << std::endl;
        } else {
            std::cout << "失败: " << message << std::endl;
        }
    });
    
    // 等待异步操作完成
    std::this_thread::sleep_for(std::chrono::seconds(3));
    return 0;
}

回调函数的优点

  1. 灵活性: 可以在运行时决定执行什么代码

  2. 解耦合: 将调用者和被调用者分离

  3. 可扩展性: 容易添加新的回调函数

  4. 异步支持: 非常适合异步编程模式

注意事项

  • 使用 std::function 比原始函数指针更安全灵活

  • 注意回调函数的生命周期管理

  • 在多线程环境中要注意线程安全问题

  • 避免过深的回调嵌套(回调地狱)

回调函数是C++中实现事件驱动编程和异步编程的重要工具,在现代C++开发中广泛应用。

相关推荐
进击的尘埃2 分钟前
Vue3 中 emit 能 await 吗?事件机制里的异步陷阱
javascript
青青家的小灰灰12 分钟前
告别 Prop Drilling:Context API 的陷阱、Reducer 模式与原子化状态库原理
前端·javascript·react.js
叶智辽14 分钟前
【Three.js后期处理】如何让你的场景拥有电影级调色
前端·three.js
进击的尘埃15 分钟前
CSS 变量 + 主题切换:从 CSS-in-JS 回归原生方案的实践之路
javascript
前端付豪16 分钟前
Nest 项目小实践之前端注册登陆
前端·node.js·nestjs
wuhen_n16 分钟前
Suspense:异步组件加载机制
前端·javascript·vue.js
wuhen_n16 分钟前
Teleport:渲染到任意DOM节点
前端·javascript·vue.js
进击的尘埃17 分钟前
组合式函数的设计模式:如何写出真正可复用的 Vue3 Composables
javascript
Moment17 分钟前
想要长期陪伴你的助理?先从部署一个 OpenClaw 开始 😍😍😍
前端·后端·github
前端Hardy17 分钟前
别再用 $emit 满天飞了!Vue 3 组件通信的 4 种正确姿势,第 3 种 90% 的人不知道
前端·vue.js·面试