(原创)C++11改进我们的程序之简化我们的程序(六)
这次要讲的内容是:c++11中的lamda表达式。
lamda表达式是我最喜欢的一个c++11特性之一,在我的代码中随处可见它的身影,其实在c#3.5中就引入了lamda,java中至今还没引入,要等明年的java8中才有lamda表达式,lamda来源于函数式编程的概念,也是现代编程语言的一个特点。c++11这次终于把lamda加进来了。也许有人要问,lamda有什么好处,为什么要加到c++中来?lamda表达式有这些优点:
- 声明式编程风格:就地匿名定义你的目标函数或函数对象,不需要额外的写一个命名函数或者函数对象。以更直接的方式去写程序,好的可读性和可维护性。
- 简洁:不需要额外再写一个函数或者函数对象,避免了代码膨胀和功能分散,让开发者更加集中精力在手边的问题,同时也获取了更高的生产率。
- 在需要的时间和地点实现谓词,使我们的程序更灵活。
lamda表达式定义了一个匿名函数,并且可以捕获一定范围内的变量。lamda表达式的语法形式:
[ capture ] ( params ) -> ret { body }
如果没有返回值也没有参数,可以这样简写:
[](){ body }或者[]{ body }
[] (int x) -> { return x; }
auto fun = [](){cout<<"test"}
fun();
lamda表达式可以截取一定范围内的变量:
[] 不截取任何变量
[&} 截取外部作用域中所有变量,并作为引用在函数体中使用
[=] 截取外部作用域中所有变量,并拷贝一份在函数体中使用
[=, &foo] 截取外部作用域中所有变量,并拷贝一份在函数体中使用,但是对foo变量使用引用
[bar] 截取bar变量并且拷贝一份在函数体重使用,同时不截取其他变量
[this] 截取当前类中的this指针。如果已经使用了&或者=就默认添加此选项。截取this的目的是可以在lamda中使用当前类的成员函数和成员变量。
声明式编程风格,简洁的代码
就地定义匿名函数,不再需要定义函数对象,大大简化了标准库算法的调用。比如c++11之前我们要调用for_each函数将vector中偶数打印出来,可能要这样写:
struct CountEven
{
CountEven(int& count) : m_count(count)
{
} void operator()(int val)
{
if(val/==)
{
m_count++;
}
} private:
int m_count;
}; int evenCount=;
for_each(v.begin(),v.end(),CountEven(evenCount));
cout<<"偶数有 "<<evenCount<<" 个"<<endl;
用lamda表达式写:
for_each(v.begin(),v.end(), [&evenCount](const int& val)
{
if(val/==)
{
evenCount++;
}
});
不再需要定义函数对象,而且更简洁明了,可读性和维护性更好,开发效率也更高。
在需要的时间和地点实现谓词,使我们的程序更灵活。
上一篇博文中的那个计算集合中大于5小于10的函数用Lamda更简洁,之前用bind的写法:
//查找大于10的元素的个数
int count = count_if(coll.begin(), coll.end(), bind1st(less<int>(), ));
//查找小于10的元素的个数
int count = count_if(coll.begin(), coll.end(), bind2nd(less<int>(), )); //查找大于5小于10的元素的个数
auto f = bind(std::logical_and<bool>(), bind(std::greater<int>(),_1,), bind(std::less_equal<int>(),_1,));
int count = count_if(coll.begin(), coll.end(), f);
用lamda的写法:
//查找大于10的元素的个数
int count = count_if(coll.begin(), coll.end(), [](int x){return x>;});
//查找小于10的元素的个数
int count = count_if(coll.begin(), coll.end(), [](int x){return x<;});
//查找大于5小于10的元素的个数
int count = count_if(coll.begin(), coll.end(), [](int x){return x>&&x<;});
孰优孰劣,一眼便知,这就是生产力啊。
lamda捕获参数
stuct A
{
A():m_cout(){}
bool Add(int x, int y)
{
return x>y;
} void TestLamda()
{
vector<int> v = {,,,};
sort(v.begin(), v.end(), [this](int x, int y){return Add(x,y);}); //调用成员函数,需要捕获this才可以 int z = ;
sort(v.begin(), v.end(), [this,z](int x, int y) //z按值捕获
{
m_count = ; //给成员变量赋值
return Add(x,y);
}); //调用成员函数,需要捕获this才可以 sort(v.begin(), v.end(), [this,&z](int x, int y) //z按引用捕获
{
z = count+; //给z赋值
return Add(x,y);
}); //以引用方式传递作用域内所有可见的局部变量(包括this)
for_each(v.begin(), v.end(), [=](int x){cout<<z+x<<endl;}); //以引用方式传递作用域内所有可见的局部变量(包括this)
for_each(v.begin(), v.end(), [&](int x){z++; cout<<z+x<<endl;}); //部分变量按引用传递
for_each(v.begin(), v.end(), [this, &z](int x){z++; cout<<z+x<<endl;});
} int m_cout;
};
lamda能捕获变量很有用,一些外面的参数可以根据需要按值或者引用传入闭包。我甚至觉得lamda表达式几乎可以取代function, 可以通过捕获对象并调用对象的成员函数的方式取代bind成员函数的方式。例如:
用bind和function的写法:
A a;
auto f = std::bind(&A::Add, &a, std::placehoders::_1,std::placehoders::_2);
int x = , y=;
auto ret = f(x, y);
用lamda的写法:
auto f = [&a](int x, int y){return a.Add(x, y);};
auto ret = f(x, y);
效果是一样的,代码还更简洁了。不过不能完全替代function的原因是lamda表达式不能作为成员变量,还有有些标准库的和boost的方法不支持lamda,还需要function出场。
c++11引入函数式编程的概念中的lamda,让我们的代码更简洁,更灵活,也更强大,并提高了开发效率,提高了可维护性。再次说一声lamada真是个好东西!
c++11 boost技术交流群:296561497,欢迎大家来交流技术。
(原创)C++11改进我们的程序之简化我们的程序(六)的更多相关文章
- (原创)C++11改进我们的程序之简化我们的程序(八)
本次要讲的是如何通过泛型函数来简化我们的程序. 泛型函数除了之前介绍的一些优点外还有两个重要的优点 1.消除重复逻辑,提高程序的内聚性和健壮性 泛型函数在某种程度上用来弥补泛型类型的不足.通过泛型类型 ...
- (原创)C++11改进我们的程序之简化我们的程序(二)
这次要讲的是:C++11如何通过组合函数来简化我们的程序.关于组合函数,大家可能对这个概念有点陌生.组合函数是将N个一元函数组成一种更复杂的函数,每个函数的返回值作为参数传给下一个函数,直到传到最后一 ...
- C++11改进我们的程序之简化我们的程序1
C++11改进我们的程序之简化我们的程序(一) C++11在很多方面可以简化我们的程序开发,我会在“简化我们的程序”这一系列的博文中一一讲到,敬请关注.这次要讲的是:C++11如何通过获取函数模板的返 ...
- (原创)C++11改进我们的程序之简化我们的程序(四)
这次要讲的是:c++11统一初始化.统一begin()/end()和for-loop循环如何简化我们的程序 初始化列表 c++11之前有各种各样的初始化语法,有时候初始化的时候还挺麻烦,比较典型的如v ...
- (原创)C++11改进我们的程序之简化我们的程序(三)
这次要讲的是:C++11如何通过auto.decltype和返回值后置来简化我们的程序. auto和c#中的var类似,都是在初始化时自动推断出数据类型.当某个变量的返回值难于书写时,或者不太确定返回 ...
- (原创)C++11改进我们的程序之简化我们的程序(一)
C++11在很多方面可以简化我们的程序开发,我会在“简化我们的程序”这一系列的博文中一一讲到,敬请关注.这次要讲的是:C++11如何通过获取函数模板的返回值类型来简化我们的程序.在谈到简化之前,我们先 ...
- (原创)c++11改进我们的模式之改进代理模式,实现通用的AOP框架
c++11 boost技术交流群:296561497,欢迎大家来交流技术. 本次要讲的时候如何改进代理模式,具体来说是动态代理模式,动态代理模式一般实现AOP框架,不懂AOP的童鞋看这里.我前面的博文 ...
- (原创)c++11改进我们的模式之改进命令模式
模式虽然精妙,却难完美,比如观察者模式中观察者生命周期的问题:比如访问者模式中循环依赖的问题等等:其它很多模式也存在这样那样的一些不足之处,如使用场景受限.实现复杂.不够简洁.不够通用等.但我觉得不足 ...
- (原创)c++11改进我们的模式之改进访问者模式
本次讲c++11改进我们的模式之改进访问者模式 访问者模式是GOF23个设计模式中比较复杂的模式之一,但是它的功能也很强大,非常适合稳定的继承层次中对象的访问,可以在不修改被访问对象的情况下,动态添加 ...
- (原创)C++11改进我们的程序之简化我们的程序(七)
这次要讲的内容是:c++11中的tuple(元组).tuple看似简单,其实它是简约而不简单,可以说它是c++11中一个既简单又复杂的东东,关于它简单的一面是它很容易使用,复杂的一面是它内部隐藏了太多 ...
随机推荐
- centos6.5官方dvd做本地yum
问题描述: 一切都搞定了,就是yum makecache 出现文件404,再目录看了也不对没后缀...(这不扯淡吗,rhel的dvd是可以直接使用的,难道企业版与社区版的区别??) 日志记录 [roo ...
- 转:Ogre的SceneManager分析
SceneManager分析 场景管理主要工作包括以下几点: 1.可移动.不可移动和可渲染物体的创建删除. 2.场景查询. 3.渲染队列. 4.动态阴影. 一. 场景对象创建 场景中的所有对象,包括可 ...
- PL/SQL报无效的窗口句柄的解决办法
在远程服务器上使用pl sql developer查询oralce数据库的时候,遇到很长的文本变量想点开小窗口看下具体内容, 但系统弹窗提示“无效的窗口句柄”,听同事介绍原来需要开启一个windows ...
- 怎样查看SSL证书的有效期?自动续期是否生效?
前面一篇教程教大家如何能够把网站的 HTTPS 的 SSL 证书自动续期.料神米课的学员动手能力都很强,已经很多都成功把证书续期了.但怎么看证书续期是否成功了呢? 使用火狐 firefox 浏览器就可 ...
- Java爬虫框架调研
Python中大的爬虫框架有scrapy(风格类似django),pyspider(国产python爬虫框架). 除了Python,Java中也有许多爬虫框架. nutch apache下的开源爬虫程 ...
- 【RS】Matrix Factorization Techniques for Recommender Systems - 推荐系统的矩阵分解技术
[论文标题]Matrix Factorization Techniques for Recommender Systems(2009,Published by the IEEE Computer So ...
- 【Java】解析Java对XML的操作
目录结构: contents structure [+] 什么是XML 解析XML 使用DOM解析 使用SAX解析 使用PULL解析 使用dom4j解析xml dom4j的部分API 打印一个XML文 ...
- Windows 10安装DockerToolBox失败处理方法
指令运行报错: Windows 10安装DockerToolBox失败处理方法:升级Windows 10. (注意:只有Windows10 专业版才支持升级,Server和企业版无效)
- 【struts2】值栈(后篇)
在值栈(前篇)我们学习了值栈的基本知识,接下来,来看看在程序中具体如何使用值栈. 1 ActionContext的基本使用 1.1 如何获取? 要获取ActionContext有两个基本的方法,如果在 ...
- 黑客公布2012年最弱智密码Top25(转)
尽管弱密码对安全性的危害大家都知道,但是仍然有很多网民使用超弱密码.日前,黑客公布了一份密码文档,列出了今年最弱智密码. 根据 SplashData 公布的“年度最弱智密码 Top25”,和去年一样, ...