博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C++学习:lambda表达式入门
阅读量:6198 次
发布时间:2019-06-21

本文共 6207 字,大约阅读时间需要 20 分钟。

引言:lambda表达式在STL应用中可以让我们起到十分便捷的功能,我们看下微软的解释。

When you write code, you probably use function pointers and function objects to solve problems and perform calculations, especially when you use . Function pointers and function objects have advantages and disadvantages—for example, function pointers have minimal syntactic overhead but do not retain state within a scope, and function objects can maintain state but require the syntactic overhead of a class definition.

A lambda combines the benefits of function pointers and function objects and avoids their disadvantages. Like a function objects, a lambda is flexible and can maintain state, but unlike a function object, its compact syntax doesn't require a class definition. By using lambdas, you can write code that's less cumbersome and less prone to errors than the code for an equivalent function obj。

当我们写代码的时候,可以使用函数指针或者函数对象去解决问题和计算,尤其在使用STL的内置算法时,不过函数指针都有明显的优劣势,举个例子,函数指针有简洁的语法但是(不包含状态一定范围内),函数对象是包含状态,但是需要专门去定义一个类并重载函数调用操作符。

 

This illustration maps the grammar to an example:

Structural elements of a lambda expression

  1. lambda-introducer (Also known as the capture clause)&捕获全部  &para 捕获特定参数 =copy

  2. lambda declarator (Also known as the parameter list)输入参数

  3. mutable (Also known as the mutable specification) 是否可以修改 const属性的参数

  4. exception-specification (Also known as the exception specification) 异常

  5. trailing-return-type (Also known as the return type) 返回值类型

  6. compound-statement (Also known as the lambda body) 函数体

 

下面引入两个例子,看一下lambda表达式带来的好处,两个的demo实现的是相同的功能;

 

1.利用函数对象实现,必须要定义一个类并重载call-function符号

1 // even_functor.cpp 2 // compile with: /EHsc 3 #include 
4 #include
5 #include
6 using namespace std; 7 8 class FunctorClass 9 {10 public:11 // The required constructor for this example.12 explicit FunctorClass(int& evenCount)13 : m_evenCount(evenCount) { }14 15 // The function-call operator prints whether the number is16 // even or odd. If the number is even, this method updates17 // the counter.18 void operator()(int n) const {19 cout << n;20 21 if (n % 2 == 0) {22 cout << " is even " << endl;23 ++m_evenCount;24 } else {25 cout << " is odd " << endl;26 }27 }28 29 private:30 // Default assignment operator to silence warning C4512.31 FunctorClass& operator=(const FunctorClass&);32 33 int& m_evenCount; // the number of even variables in the vector.34 };35 36 37 int main()38 {39 // Create a vector object that contains 10 elements.40 vector
v;41 for (int i = 1; i < 10; ++i) {42 v.push_back(i);43 }44 45 // Count the number of even numbers in the vector by 46 // using the for_each function and a function object.47 int evenCount = 0;48 for_each(v.begin(), v.end(), FunctorClass(evenCount));49 50 // Print the count of even numbers to the console.51 cout << "There are " << evenCount52 << " even numbers in the vector." << endl;53 }
输出: 1 is odd2 is even3 is odd4 is even5 is odd6 is even7 is odd8 is even9 is oddThere are 4 even numbers in the vector 看看简洁的lambda表达式如何实现同样的功能。
1 // even_lambda.cpp 2 // compile with: cl /EHsc /nologo /W4 /MTd 3 #include 
4 #include
5 #include
6 using namespace std; 7 8 int main() 9 {10 // Create a vector object that contains 10 elements.11 vector
v;12 for (int i = 1; i < 10; ++i) {13 v.push_back(i);14 }15 16 // Count the number of even numbers in the vector by 17 // using the for_each function and a lambda.18 int evenCount = 0;19 for_each(v.begin(), v.end(), [&evenCount] (int n) {20 cout << n;21 if (n % 2 == 0) {22 cout << " is even " << endl;23 ++evenCount;24 } else {25 cout << " is odd " << endl;26 }27 });28 29 // Print the count of even numbers to the console.30 cout << "There are " << evenCount 31 << " even numbers in the vector." << endl;32 }

Output:

1 is odd2 is even3 is odd4 is even5 is odd6 is even7 is odd8 is even9 is oddThere are 4 even numbers in the vector. Summery: 看得出lambda表达式代码的好处十分的明显,使代码变得简洁,语义变的清晰。代码不易膨胀,出bug的概率也变小了。下面我们介绍下lambda最基本的语法 Caputre的规则
struct S { void f(int i); };void S::f(int i) {    [&, i]{};    // OK    [&, &i]{};   // ERROR: i preceded by & when & is the default    [=, this]{}; // ERROR: this when = is the default    [i, i]{};    // ERROR: i repeated}

上下文的capurue要注意一些规则, [&]是获取全部reference而不是单个变量的reference, [&para]才是获取单个变量的reference,

=同理,只不过是上下文全部以copy的方式进行传递。

 

lambda :body的规则

// captures_lambda_expression.cpp// compile with: /W4 /EHsc #include 
using namespace std;int main(){ int m = 0; int n = 0; [&, n] (int a) mutable { m = ++n + a; }(4); cout << m << endl << n << endl;}

 

lambda表达式的body中可以包含方法或者函数,而这些函数是可以访问lambda所captrue的变量的。

 The body of both an ordinary function and a lambda expression can access these kinds of variables:

  • Parameters

  • Locally-declared variables

  • Class data members, when declared inside a class and this is captured

  • Any variable that has static storage duration—for example, global variables

以上4中变量都是可以访问的。

In addition, a lambda expression can access variables that it captures from the enclosing scope. A variable is explicitly captured if it appears in the capture clause of the lambda expression. Otherwise, the variable is implicitly captured. The body of the lambda expression uses the default capture mode to access variables that are implicitly captured.

The following example contains a lambda expression that explicitly captures the variable n by value and implicitly captures the variable m by reference:

值得注意的是,lambda表达式可以访问不同的变量从captures,一个变量可以显式的从捕获中获取,或者通过隐式捕获,body中可以使用默认的捕获方式去访问隐式捕获的变量 下面这个例子就是看到显式的捕获n,和隐式的捕获m的引用。
1 // captures_lambda_expression.cpp 2 // compile with: /W4 /EHsc  3 #include 
4 using namespace std; 5 6 int main() 7 { 8 int m = 0; 9 int n = 0;10 [&, n] (int a) mutable { m = ++n + a; }(4);11 cout << m << endl << n << endl;12 }

Output:

50
更多详情请看 https://msdn.microsoft.com/en-us/library/dd293603.aspx

转载于:https://www.cnblogs.com/DLzhang/p/4645978.html

你可能感兴趣的文章
CVE-2017-7494 Linux Samba named pipe file Open Vul Lead to DLL Execution
查看>>
cocos2dx 几个精灵依照顺序播放动画解决方法
查看>>
mysql source 命令导入不了sql!
查看>>
stm32 usart的几种通信模式
查看>>
MyReport报表系统v1.2公布
查看>>
TCP/IP各层协议数据格式
查看>>
手工清理win7系统C盘的技巧
查看>>
eclipse 灵活使用makefile来编译C/C++
查看>>
jquery.cookie() 方法的使用(读取、写入、删除)
查看>>
Java -- JDBC 学习--处理Blob
查看>>
一个执念重度患者的自白
查看>>
CO-PRIME(初探 莫比乌斯)NYOJ1066(经典)gcd(a,b)=1
查看>>
bzoj1497【NOI2006】最大获利
查看>>
基于Jersey使用Session
查看>>
【hoj】2160 bin packing 二分、贪心
查看>>
为OLED屏添加GUI支持2:2D图形库
查看>>
IOS 数据存储之 SQLite具体解释
查看>>
利用js在文本框末尾获得焦点
查看>>
[Phonegap+Sencha Touch] 移动开发19 某些安卓手机上弹出消息框 点击后不消失的解决的方法...
查看>>
Struts2概述及与Struts1的对照
查看>>