转自:https://msdn.microsoft.com/zh-cn/library/dd293603.aspx
本文演示了 lambda 表达式的语法和结构化元素。有关 lambda 表达式的说明,请参阅 C++ 中的 Lambda 表达式。
你编写代码时,尤其是使用 STL 算法时,可能会使用函数指针和函数对象来解决问题和执行计算。函数指针和函数对象各有利弊。例如,函数指针具有最低的语法开销,但不保持范围内的状态,函数对象可保持状态,但需要类定义的语法开销。
lambda 结合了函数指针和函数对象的优点并避免其缺点。lambda 与函数对象相似的是灵活并且可以保持状态,但不同的是其简洁的语法不需要显式类定义。 使用lambda,相比等效的函数对象代码,您可以写出不太复杂并且不容易出错的代码。
下面的示例比较lambda和函数对象的使用。 第一个示例使用 lambda 向控制台打印 vector 对象中的每个元素是偶数还是奇数。第二个示例使用函数对象来完成相同任务。
此示例将一个 lambda 传递给 for_each 函数。该 lambda 打印一个结果,该结果指出 vector 对象中的每个元素是偶数还是奇数。
// even_lambda.cpp // compile with: cl /EHsc /nologo /W4 /MTd #include <algorithm> #include <iostream> #include <vector> using namespace std; int main() { // 创建一个包含十个元素的vector对象 vector<int> v; for (int i = 1; i < 10; ++i) { v.push_back(i); }
// 通过使用for_each函数和lambda表达式,统计vector中偶数个数 int evenCount = 0; for_each(v.begin(), v.end(), [&evenCount] (int n) { cout << n; if (n % 2 == 0) { cout << " is even " << endl; ++evenCount; } else { cout << " is odd " << endl; } }); // 将偶数个数打印到终端 cout << "There are " << evenCount << " even numbers in the vector." << endl; }
1 为奇数 2 为偶数 3 为奇数 4 为偶数 5 为奇数 6 为偶数 7 为奇数 8 为偶数 9 为奇数 在向量中存在 4 个偶数。
在此示例中,for_each 函数的第三个参数是一个lambda。 [&evenCount] 部分指定表达式的捕获子句,(int n) 指定参数列表,剩余部分指定表达式的主体。
有时 lambda 过于庞大,无法在上一示例的基础上大幅度扩展。下一示例使用函数对象(而非 lambda)以及 for_each 函数,以产生与示例 1 相同的结果。两个示例都在 vector 对象中存储偶数的个数。为保持运算的状态,FunctorClass 类通过引用存储 m_evenCount 变量作为成员变量。为执行该运算,FunctorClass 实现函数调用运算符 operator()。Visual C++ 编译器生成的代码与示例 1 中的 lambda 代码在大小和性能上相差无几。对于类似本文中示例的基本问题,较为简单的 lambda 设计可能优于函数对象设计。但是,如果你认为该功能在将来可能需要重大扩展,则使用函数对象设计,这样代码维护会更简单。
有关 operator() 的详细信息,请参阅函数调用 (C++)。有关 for_each 函数的详细信息,请参阅 for_each。
// even_functor.cpp // compile with: /EHsc #include <algorithm> #include <iostream> #include <vector> using namespace std; class FunctorClass { public: // 对于该例子需要一个构造函数 explicit FunctorClass(int& evenCount) : m_evenCount(evenCount) { } // 重载函数调用运算符,用来打印数字是奇是偶。如果是偶数,计数器增1。 void operator()(int n) const { cout << n; if (n % 2 == 0) { cout << " is even " << endl; ++m_evenCount; } else { cout << " is odd " << endl; } } private: // 重载运算符 FunctorClass& operator=(const FunctorClass&); int& m_evenCount; //vector中的偶数个数 }; int main() { // 创建一个包含十个元素的vector对象 vector<int> v; for (int i = 1; i < 10; ++i) { v.push_back(i); } // 通过使用for_each函数和类构造函数,统计vector中偶数个数 int evenCount = 0; for_each(v.begin(), v.end(), FunctorClass(evenCount)); // 将偶数个数打印到终端 cout << "There are " << evenCount << " even numbers in the vector." << endl; }
1 为奇数 2 为偶数 3 为奇数 4 为偶数 5 为奇数 6 为偶数 7 为奇数 8 为偶数 9 为奇数在向量中存在 4 个偶数。