在 C++11 标准带来的诸多革命性特性中,简化代码编写与统一可调用对象管理是两大核心目标。Lambda 表达式解决了传统仿函数定义繁琐的痛点,让局部场景下的自定义逻辑能以更简洁的匿名函数形式实现;可变参数模板打破了模板参数数量固定的限制,为 STL 容器和通用函数设计提供了灵活的参数处理能力;而 function 包装器与 bind 函数,则进一步整合了函数指针、仿函数、lambda 等不同类型的可调用对象,实现了统一管理与参数适配。
这些特性并非孤立存在。Lambda 的底层依赖仿函数实现,可变参数模板为 emplace 系列接口提供了技术支撑,function 与 bind 则基于前两者的特性,解决了不同可调用对象类型不统一的问题。本文将从实际开发需求出发,先讲解 Lambda 表达式的语法规则与捕获逻辑,再深入可变参数模板的展开方法与应用场景,最后通过 function 包装器与 bind 函数的实例,展示如何统一管理各类可调用对象。
Lambda 表达式
在实际开发中,如果一个类有很多成员变量需要排序,但每次比较的逻辑不一样,且不想每次都写一个独立的类去比较,Lambda 表达式就是很好的解决方案。它是局部的匿名函数对象,可以直接用在需要仿函数的地方(如 sort),其底层其实就是仿函数。
需要注意的是,Lambda 表达式不存在重载的说法。即使两个 Lambda 看起来一样,它们也是不同的类型,不能相互赋值。例如:
auto f1 = [](){};
auto f2 = [](){};
// f1 = f2; // 编译错误,类型不同
语法结构
Lambda 的基本格式如下:
[捕捉列表](参数列表)(mutable)-> 返回值类型 { 函数体 };
- 捕捉列表:用于捕获表达式所在域的局部变量。全局域里的变量不用捕获也能用。如果参数列表不传参,
()可以省略。 - mutable:取消捕获列表里东西默认的 const 性。注意,加了 mutable 修饰符时,参数列表不能省略。如果捕获的东西本身是 const,加了 mutable 也变不了。
- 返回值类型:一般可以不用写,编译器会推导。
捕获列表详解
如果没有使用 mutable,捕获的东西都不能被修改。
[var]:值传递方式捕捉变量 var,类型与传递过来的一模一样。[=]:值传递方式捕获所有父作用域中的变量(包括 this)。[&var]:引用传递捕捉变量 var。[&]:引用传递捕捉所有父作用域中的变量(包括 this)。[&, x]:表示只有 x 是值传递,其他都是引用。
注意:捕捉列表不允许变量重复传递,否则会导致编译错误,例如 [=, a] 是不行的,因为 a 被捕获了两次。
可调用对象的存储
Lambda 表达式作为可调用对象之一,可以通过以下四种方式存储:函数指针、仿函数、Lambda 表达式、以及使用包装器放入容器中。
可变参数模板
可变参数模板允许模板参数具有不确定的数目。虽然日常开发中使用频率不如 Lambda 高,但在泛型编程中非常关键。
template<class... Args>
void ShowList(Args... args) {
// Args 表示一个包含零个或多个类型的参数包
// args 也是参数包,里面可以是 0 或多个参数
}


