std::function 和 std::bind 是 C++ 中处理可调用对象的核心工具,前者提供统一接口封装函数、Lambda、成员函数等,后者支持参数绑定与占位符灵活适配,二者结合广泛用于回调机制与事件系统,提升代码复用性与灵活性。

在C++中,std::function 和 std::bind 是处理可调用对象的强大工具,它们让函数指针、lambda表达式、成员函数、仿函数等统一接口调用成为可能。掌握它们的使用,能显著提升代码的灵活性和复用性。
std::function:通用可调用对象包装器
std::function 是一个模板类,定义在 <functional> 头文件中,可以封装任何可调用对象,只要其签名匹配。
基本语法:
// 声明一个接受int返回void的函数对象
std::function<void(int)> func;
示例:
立即学习“C++免费学习笔记(深入)”;
#include <functional>
#include <iostream>
void simple_print(int x) {
std::cout << “Value: ” << x << std::endl;
}
int main() {
std::function<void(int)> f = simple_print;
f(42); // 输出: Value: 42
// 绑定 lambda
f = [](int x) { std::cout << “Lambda: ” << x << std::endl; };
f(100); // 输出: Lambda: 100
}
它支持多种可调用类型:
– 普通函数
– Lambda 表达式
– 函数对象(仿函数)
– 成员函数指针(需配合对象或指针使用)
– std::bind 的结果
std::bind:灵活绑定参数和对象
std::bind 可以将函数的参数部分提前绑定,生成一个新的可调用对象,常用于适配回调、延迟调用或固定某些参数。
语法:
std::bind(func, arg1, arg2, …)
占位符 _1, _2, … 来表示运行时传入的参数,定义在 std::placeholders 名称空间中。
常见用途:
- 绑定成员函数到具体对象
- 固定部分函数参数
- 调整参数顺序
示例:
立即学习“C++免费学习笔记(深入)”;
#include <functional>
#include <iostream>
using Namespace std::placeholders; // 必须引入
Struct Calculator {
int add(int a, int b) { return a + b; }
};
int main() {
Calculator calc;
// 绑定成员函数和对象实例
auto bound_add = std::bind(&Calculator::add, &calc, _1, _2);
std::cout << bound_add(3, 5) << std::endl; // 输出: 8
// 固定第一个参数
auto add_10 = std::bind(&Calculator::add, &calc, 10, _1);
std::cout << add_10(7) << std::endl; // 输出: 17
}
结合使用:实现回调和事件系统
std::function 和 std::bind 结合,非常适合实现回调机制。
例如:
class EventManager {
public:
using Callback = std::function<void(int)>;
void set_callback(Callback cb) { callback = cb; }
void trigger(int value) { if (callback) callback(value); }
private:
Callback callback;
};
struct Handler {
void on_event(int v) { std::cout << “Handled: ” << v << std::endl; }
};
int main() {
EventManager mgr;
Handler h;
// 绑定成员函数作为回调
mgr.set_callback(std::bind(&Handler::on_event, &h, _1));
mgr.trigger(99); // 输出: Handled: 99
}
这种模式广泛用于GUI、异步任务、观察者模式等场景。
基本上就这些。std::function 提供统一调用接口,std::bind 实现参数灵活绑定,两者配合能让C++的可调用对象处理更优雅。注意 bind 的占位符作用域和对象生命周期管理,避免悬空引用。现代C++中,lambda 有时更简洁,但 bind 在需要重用参数绑定逻辑时仍有优势。


