std::function 可包装任意可调用对象,实现统一接口;std::bind 能绑定部分参数并重排顺序,二者结合常用于回调、事件系统等场景,提升灵活性。

在C++函数式编程中,std::function 和 std::bind 是两个非常实用的工具,它们让函数对象、回调机制和延迟调用变得更灵活。下面通过具体例子说明它们的基本用法和常见技巧。
std::function:统一可调用对象类型
std::function 是一个通用的函数包装器,可以保存、复制和调用任何可调用的目标,比如普通函数、lambda表达式、函数对象或成员函数指针。
基本语法:
std::function
立即学习“C++免费学习笔记(深入)”;
示例:
- 包装普通函数
#include
#include
double add(double a, double b) {
return a + b;
}
std::functionfunc = add;
std::cout << func(2.5, 3.0) << std::endl; // 输出 5.5
- 绑定 lambda 表达式
std::functionmax_func = [](int a, int b) {
return a > b ? a : b;
};
std::cout << max_func(4, 7) << std::endl; // 输出 7
- 用于回调函数
将 std::function 作为参数传递,实现灵活的回调机制:
void process_data(std::functioncallback) {
for (int i = 0; i < 5; ++i) {
callback(i);
}
}
process_data([](int x) { std::cout << "处理: " << x << std::endl; });
std::bind:绑定参数生成可调用对象
std::bind 可以把函数的部分参数预先绑定,生成一个新的可调用对象,常用于适配函数签名或固定某些参数。
基本语法:
std::bind(函数名, 参数1, 参数2...)
占位符 std::placeholders::_1, _2 等表示运行时传入的参数位置。
- 绑定部分参数
例如,有一个三参数函数,想固定前两个参数:
#include
int multiply(int a, int b, int c) {
return a * b * c;
}
auto bind_first_two = std::bind(multiply, 2, 3, std::placeholders::_1);
std::cout << bind_first_two(4) << std::endl; // 相当于 multiply(2, 3, 4) → 24
- 调整参数顺序
使用占位符重新排列参数顺序:
auto reorder = std::bind(multiply, std::placeholders::_2, std::placeholders::_1, 5);
std::cout << reorder(2, 3) << std::endl; // 相当于 multiply(3, 2, 5) → 30
- 绑定成员函数
必须传入对象实例(或指针)作为第一个参数:
struct Calculator {
int add(int a, int b) { return a + b; }
};
Calculator calc;
auto add_bind = std::bind(&Calculator::add, &calc, std::placeholders::_1, std::placeholders::_2);
std::cout << add_bind(3, 4) << std::endl; // 输出 7
结合使用:std::function + std::bind
通常会将 std::bind 的结果赋给 std::function,以便统一管理类型或作为回调传参:
std::functionfunc = std::bind(multiply, 2, std::placeholders::_1, 10);
std::cout << func(3) << std::endl; // 相当于 multiply(2, 3, 10) → 60
这种组合特别适合事件系统、线程任务或需要注册不同调用形式的场景。
基本上就这些。std::function 提供了类型擦除的调用接口,std::bind 则增强了函数的复用性和灵活性。虽然 C++11 之后 lambda 更常用,但在需要参数重排或长期保存绑定逻辑时,std::bind 仍有其价值。不复杂但容易忽略的是占位符的作用域和绑定对象的生命周期管理。










