lambda表达式

函数式编程的一个语法,有如下优点: 
(1)声明式编程风格:就地匿名定义目标函数或函数对象,不需要额外写一个命名函数或者函数对象。以更直接的方式写程序,好的可读性和可维护性。 
(2)简洁:不需要额外再写一个函数或者函数对象,避免了代码膨胀和功能分散。 
(3)在需要的时间和地点实现功能闭包,使程序更灵活。

lambda表达式基本概念和用法

lambda表达式定义了一个匿名函数,并且可以捕获一定范围的变量。lambda表达式的语法如下: 
[capture] (params) opt -> ret {body;} 
其中,capture为捕获列表;params是参数表;opt为函数选项;ret为返回值类型;body是函数体。捕获列表是将lambda外部的一些变量引入lambda内部,可以为值类型或引用类型。

    auto f = [] (int a) -> int {return a + 1;};
std::cout << f(1) << std::endl;

或者省略返回值类型(c++11可以进行类型推导),直接写成

    auto f = [](int a){return a+1;};
std::cout << f(1) << std::endl;
捕获列表

lambda表达式可以通过捕获列表捕获一定范围内的变量:

[] 不捕获任何变量 
[&] 捕获外部作用域中的所有变量,并作为引用在函数体内使用(按引用捕获) 
[=] 捕获外部作用域内的所有变量,并作为副本在函数体内使用(按值捕获) 
[=, &foo] 按值捕获外部作用域中的所有变量,并按引用捕获foo变量 
[bar] 按值捕获bar变量,同时不捕获其他变量 
[this] 捕获当前类中的this指针,让lambda表达式拥有和当前类成员函数同样的访问权限。如果已经使用了=或者&,就默认添加此项。捕获this的目的是可以在lambda中使用当前类的成员函数和成员变量。

    int a = 0, b = 1;
auto f1 = []{return a;} ;//错误,没有捕获外部变量
auto f2 = [&]{return a++;};//捕获外部变量的引用,并且可以实现将a++
auto f3 = [=]{return a;};
auto f4 = [=]{return a++;};//错误,a是一个副本,并且为const类型,无法修改
auto f5 = [a, &b]{return a + (b++);}//可以,按值捕获a,并且按引用捕获b

需要注意,默认情况下,lambda表达式无法修改通过复制方式捕获的外部变量。如果希望修改这些外部变量,可以通过引用方式进行捕获,或者通过显式指明lambda表达式为mutable类型:

    int a = 0;
auto f1 = [=]{return a++;};//错误
auto f2 = [=]()mutable{return a++;} //需要注意,mutable修饰的lambda即使没有参数,也要书写参数列表
int main(){
int a = 1;
auto f = [=]{return a+1; };
cout << f() << endl; //输出2
cout << "a = " << a << endl; //a = 1
auto f1 = [=]()mutable{return a++; };
cout << "a = " << a << endl; //a = 1
cout << f() << endl; //输出2 ????
cout << "a = " << a << endl; //a = 1
return 0;
}

以及需要注意lambda的延迟调用

  1. int a = 0;
  2. auto f = [=]{return a;};
  3. a += 1;
  4. std::cout << f() << std::endl; //输出的a为之前的a

lambda表达式按值捕获外部的变量,在捕获的一瞬间,这些值就被复制到f中。

lambda表达式和闭包

在c++11中,lambda为闭包类型,是一个特殊的、匿名的非union的类类型。可以认为lambda是一个带有operator()的类,即仿函数。因此可以使用std::function和std::bind来存储和操作lambda表达式。

    std::function<int(int)> f1 = [] (int a){return a;};
std::function<int(void)> f2 = std::bind([](int a){return a;}, 123);

对于没有捕获任何变量的lambda表达式,还可以被转换成一个普通的函数指针:

    using func_t = int(*)(int);
func_t f = [](int a){return a;};
f(123);

可以认为lambda是一个带有operator()的类,即仿函数。按照c++的标准,lambda表达式的operator()默认是const的,因此在按值捕获的时候不能修改外部变量的值;而使用mutable可以取消 operator()的const。

就地声明和使用lambda表达式

int count = count_if(coll.begin(), coll.end(), [](int x){return x > 5 && x < 10;});
+

c++11——lambda表达式的更多相关文章

  1. C++11 lambda 表达式

    C++11 新增了很多特性,lambda 表达式是其中之一,如果你想了解的 C++11 完整特性,建议去这里,这里,这里,还有这里看看.本文作为 5 月的最后一篇博客,将介绍 C++11 的 lamb ...

  2. C++11 lambda 表达式解析

    C++11 新增了很多特性,lambda 表达式是其中之一,如果你想了解的 C++11 完整特性,建议去这里,这里,这里,还有这里看看.本文作为 5 月的最后一篇博客,将介绍 C++11 的 lamb ...

  3. 详解 C++11 lambda表达式

    详解 C++11 lambda表达式   lambda表达式是函数式编程的基础.咱对于函数式编程也没有足够的理解,因此这里不敢胡言乱语,有兴趣的可以自己查找相关资料看下.这里只是介绍C++11中的la ...

  4. C++11 lambda表达式学习

    lambda表达式是函数式编程的基础.咱对于函数式编程也没有足够的理解,因此这里不敢胡言乱语,有兴趣的可以自己查找相关资料看下.这里只是介绍C++11中的lambda表达式自己的认识.这里有参考文档h ...

  5. C++ 11 Lambda表达式

    C++11的一大亮点就是引入了Lambda表达式.利用Lambda表达式,可以方便的定义和创建匿名函数.对于C++这门语言来说来说,“Lambda表达式”或“匿名函数”这些概念听起来好像很深奥,但很多 ...

  6. C++11 lambda表达式是如何实现的?

    lambda表达式是如何实现的呢? 其实是编译器为我们了创建了一个类,这个类重载了(),让我们可以像调用函数一样使用.所以,你写的lambda表达式和真正的实现,是这个样子的: 而对于捕获变量的lam ...

  7. C++ 11 Lambda表达式、auto、function、bind、final、override

    接触了cocos2dx 3.0,就必须得看C++ 11了.有分享过帖子:[转帖]漫话C++0x(四) —- function, bind和lambda.其实最后的Lambda没太怎么看懂. 看不懂没关 ...

  8. C++11 Lambda表达式简单解析

    C++11 新增了非常多特性,lambda 表达式是当中之中的一个.假设你想了解的 C++11 完整特性, 建议去http://www.open-std.org/看看新标准! 非常多语言都提供了 la ...

  9. C++11 lambda表达式(19篇C++11文章)

    C++11引入了lambda表达式,使得程序员可以定义匿名函数,该函数是一次性执行的,既方便了编程,又能防止别人的访问. Lambda表达式的语法通过下图来介绍: 这里假设我们定义了一个如上图的lam ...

随机推荐

  1. 【C#/WPF】用System.Timers.Timer计时器做浮窗广告

    需求:鼠标静止一段时间后,显示浮窗广告. 思路:界面XAML写好一个专门显示浮窗广告的Canvas,先设为不可见Visibility=”Collapsed”,然后用System.Timers.Time ...

  2. Jdbc获取oracle中guid主键

    上代码 String sql = "BEGIN insert into itil_task_plan (PLAN_CODE) values (?) returning id into ?; ...

  3. UTF-8以字节为单位对Unicode进行编码

    UTF-8以字节为单位对Unicode进行编码.从Unicode到UTF-8的编码方式如下: Unicode编码(16进制) UTF-8 字节流(二进制) 000000 - 00007F 0xxxxx ...

  4. Hive UDF IP解析(二):使用geoip2数据库自定义UDF

    开发中经常会碰到将IP转为地域的问题,所以以下记录Hive中自定义UDF来解析IP. 使用到的地域库位maxmind公司的geoIP2数据库,分为免费版GeoLite2-City.mmdb和收费版Ge ...

  5. if no 和 if not

    x = None if x is not None: print("kong") if not x is not None: print("kong") # i ...

  6. ggplot饼图

    目录: 原始图样 如何去除饼图中心的杂点 如何去除饼图旁边的标签 如何去掉左上角多出来的一横线 如何去掉图例的标题,并将图例放到上面 如何对图例的标签加上百分比 如何让饼图的小块按顺时针从大到小的顺序 ...

  7. C语言 float、double数据在内存中的存储方式

    float在内存中占4个字节(32bit),32bit=符号位(1bit)+指数位(8bit)+底数位(23bit) 指数部分 指数位占8bit,可以表示数值的范围是0-(表示0~255一共256个数 ...

  8. android listview 总结

    ScrollView与ListView冲突: public class MyExpandableListView extends ExpandableListView { public MyExpan ...

  9. selenium测试(Java)--元素操作(五)

    元素的操作有 1. 清除文本 2. 模拟按键输入 3. 单击元素 4. 返回元素尺寸 5. 获取文本 6. 获取属性值 7. 判断是否可见 8. 提交 下面通过操作新浪邮箱的注册界面的脚本来展示使用方 ...

  10. 从头认识java-18.2 主要的线程机制(7)join

    这一章节我们来讨论一下join. 1.什么是join? 阻塞当前线程,让.join的线程先做完.然后再继续当前线程 以下是api的解释: A.join,在API中的解释是,阻塞当前线程B,直到A运行完 ...