学习C++11的一些思考和心得(1):lambda,function,bind和委托
1.lambda表达式
lanbda表达式简单地来讲就是一个匿名函数,就是没有名称的函数,如果以前有接触过python或者erlang的人都比较熟悉这个,这个可以很方便地和STL里面的算法配合
std::for_each(list.begin(),list.end(),
[ &tmp,id ](struct ComingVesselInfo *info)
{
if( info->m_nShipID == id)
{
tmp = info;
return ;
}
}
);
这个是我在项目里面使用的一段代码,如果没有lambda,的代码则是这样的:
for ( auto iter = list.begin();
iter != list.end();
++iter)
{
if ( iter->m_nShipID == id)
{
tmp = iter;
}
}
从上面的代码比较并没有看出lambda有什么用,反而丧失了一点代码的可读性,但是我觉得lambda的引入不是我们可以在函数内直接很方便地写函数,而是可以很好地跟STL algorithm库配合,在我们的项目中,我很少使用到STL algorithm库,因为大部分算法函数都有下面的两个问题:
1.无法获取返回值
2.无法传入其他的参数(比如上面的for_each的例子,如果没有lambda,并没有办法把tmp和id传入)
lambda的引入解决了我在项目中使用STL的algorithm的顾虑
顺便说下,上面的auto也是C++11的新标准,就是自动推导右边数据的类型
2.function和bind以及委托
在C#里面,我们经常会看到使用委托的,C#的委托语法是这样的:
public delegate void Ticket();
这样的话,我们只要声明一个Ticket t,这个只要符合functionname()这个形式的参数,不管是static 函数还是类成员函数,都可以赋值给t,并且可以通过t()来调用该函数或者成员函数
很多学C#的程序员一看这个就简单地把委托认为是C++里面的函数指针,其实这个是错的,委托和函数指针的区别在于,委托是有状态的,更类似与闭包,而函数指针只是一个地址,是没有状态的,单纯的函数指针无法根据我们的需求去传入外部的状态(也就是无法根据需求去改变传入参数的个数和参数的类型)
在C++里面,我们只能通过这样来指向一个成员函数的指针:
int (simple:: *pfn) (int) = simple::fn;
这样地话我们才能把一个类成员函数指针指向一个类的成员函数,但是这样明显有两个缺点:
1.我们需要在函数调用的时候才能传入参数,无法在函数确定要指向哪一个类成员函数的时候指定参数
2.最重要的一点,我们无法把pfn认识指向一个functionname(int)的类成员函数,我们只能再次定义一个函数指针
在C++里面实际上也是有类似于委托的功能,利用类的成员变量来保存我们的状态,然后重载operator(),就可以达到我们所说的闭包的效果
class Add
{
public:
Add(int i,int j)
{
m_i = i;
m_j = j;
} int operator()()
{
return m_i + m_j;
} protected:
int m_i;
int m_j;
};
这样我们就可以这样使用这个Add类
Add add(1,2);
cout<<add()<<endl;
但是很明显地,这样做我们每次都需要根据需求去创建一个class,实际上并没有比我们根据需求来写函数方便地多少,还不如直接使用lambda,在C++ 11则解决了这个问题,引入了boost的function和bind
class calc
{
public:
calc(int base)
{
m_base = base;
} int Add(int j)
{
m_base += j;
return m_base;
} int sub(int j)
{
m_base -= j;
return m_base;
}
protected:
int m_base;
};
int main(int argc,char *argv[])
{
calc *c = new calc(10);
tr1::function<int ()> fun; fun = tr1::bind(&calc::Add,c,3);
cout<<fun()<<endl; tr1::function<int (int)> fun2;
fun2 = tr1::bind(&calc::sub,c,tr1::_1);
cout<<fun2(3); }
function和bind的引入解决了我们无法使用C#里面委托的问题,并且既可以提前绑定参数,又可以延后绑定参数,而且可以跟任何类成员函数配合
看起来是很方便的,但是别忘记了C++里面的一个问题,就是C++没有内存回收的功能,如果我们bind的类被释放掉了,那么会怎么样??
如果我们main函数改成下面这样调用:
calc *c = new calc(10);
tr1::function<int ()> fun; fun = tr1::bind(&calc::Add,c,3);
delete c;
c = NULL; cout<<fun()<<endl;
那么我们调用fun()的时候会发生什么事情??
我们无法得知计算得结果,因为此时的c已经被delete掉了,虽然可以依然正常调用Add函数(如果对C++比较熟悉的应该知道为什么可以正常调用Add),但是内部我们保存的"状态"不见了,也随着delete动作一起消失了,留下的是一串随机的数
但是如果我们把下面的代码改成这样:
tr1::function<int ()> fun2;
{
calc d(12);
fun2 = tr1::bind(&calc::Add,d,40);
}
fun2();
那么即使在d释放掉以后,我们依然可以正常地调用fun2
根据我的测试,在我们调用bind的时候,实际上将整个类的当前"状态"都复制了过去,而且在VS2010的平台上,我自己测试了下,调用了9次的复制构造函数,如果我们不好好处理这个,对于性能将会是巨大的损失
当时标准库的设计者也没有那么地傻,所以bind也可以传入指针:
fun2 = tr1::bind(&calc::Add,&d,40);
当然这个又涉及到类的状态保存的问题,类销毁了怎么办??
所以说C++的function和bind的再能模仿,也无法模仿到C#的委托,毕竟C++里面没有人帮我们管理内存,而C#程序员似乎不用关心这些,C++程序员永远离不开内存管理的话题
参考资料:
以boost::function和boost:bind取代虚函数
http://www.boost.org/doc/libs/1_54_0/doc/html/function.html
http://www.boost.org/doc/libs/1_54_0/libs/bind/bind.html
http://msdn.microsoft.com/zh-cn/library/vstudio/dd293608.aspx
学习C++11的一些思考和心得(1):lambda,function,bind和委托的更多相关文章
- Ext.Net学习笔记11:Ext.Net GridPanel的用法
Ext.Net学习笔记11:Ext.Net GridPanel的用法 GridPanel是用来显示数据的表格,与ASP.NET中的GridView类似. GridPanel用法 直接看代码: < ...
- SQL反模式学习笔记11 限定列的有效值
目标:限定列的有效值,将一列的有效字段值约束在一个固定的集合中.类似于数据字典. 反模式:在列定义上指定可选值 1. 对某一列定义一个检查约束项,这个约束不允许往列中插入或者更新任何会导致约束失败的值 ...
- golang学习笔记11 golang要用jetbrain的golang这个IDE工具开发才好
golang学习笔记11 golang要用jetbrain的golang这个IDE工具开发才好 jetbrain家的全套ide都很好用,一定要dark背景风格才装B 从File-->s ...
- Spring MVC 学习笔记11 —— 后端返回json格式数据
Spring MVC 学习笔记11 -- 后端返回json格式数据 我们常常听说json数据,首先,什么是json数据,总结起来,有以下几点: 1. JSON的全称是"JavaScript ...
- Python3+Selenium3+webdriver学习笔记11(cookie处理)
#!/usr/bin/env python# -*- coding:utf-8 -*-'''Selenium3+webdriver学习笔记11(cookie处理)'''from selenium im ...
- 并发编程学习笔记(11)----FutureTask的使用及实现
1. Future的使用 Future模式解决的问题是.在实际的运用场景中,可能某一个任务执行起来非常耗时,如果我们线程一直等着该任务执行完成再去执行其他的代码,就会损耗很大的性能,而Future接口 ...
- 《C++ Primer Plus》学习笔记11
<C++ Primer Plus>学习笔记11 第17章 输入.输出和文件 <<<<<<<<<<<<<< ...
- SpringMVC:学习笔记(11)——依赖注入与@Autowired
SpringMVC:学习笔记(11)——依赖注入与@Autowired 使用@Autowired 从Spring2.5开始,它引入了一种全新的依赖注入方式,即通过@Autowired注解.这个注解允许 ...
- 零基础学习前端1-1配置node及npm环境变量
零基础学习前端1-1配置node及npm环境变量 ## 1-1配置node及npm环境变量 首先:下载node 可以直接去官方网站下载 1.首先从官网下载安装包 https://nodejs.org/ ...
随机推荐
- AudioUnit,AudioQueue之争
最近在修改调试一个webrtc的问题,困扰了好久,故先记录下来,有成果了就发出来.问题1 使用webrtc源码做iOS上的VOIP通讯时(iOS侧接口用的AudioUnit),如果通话中被CS域来电打 ...
- HTML5 3D动画效果
对以前来讲,3D动画拿到网页上展示是一件非常奢侈的事情,第一是浏览器不够先进,第二是大部分只能用flash实现伪3D.HTML5的出现,让实现网页3D动画变得非常简单,当然前提是你不要再使用像IE67 ...
- HTML <meta> 标签 遇到<meta http-equiv="refresh" content="0; url=">详解
页面定期刷新,如果加url的,则会重新定向到指定的网页,content后面跟的是时间(单位秒),把这句话加到指定网页的<head></head>里一般也用在实时性很强的应用中, ...
- C++ Primer : 第十二章 : 文本查询程序
C++ Primer书上这个例子讲的很不错,写写帮助自己理解标准库和智能指针. .h 文件内容 #include <fstream> #include <iostream> # ...
- c++内置函数---7
原创博客:转载请标明出处:http://www.cnblogs.com/zxouxuewei/ 内置函数 调用函数时需要一定的时间和空间的开销.一般,函数调用包括5部分: 1.程序先执行函 数调用之前 ...
- c#部分---用结构体的题目- //请输入班级人数,输入每个人的学号,姓名,和语文分数、数学分数和英语分数(要求使用结构体)
//请输入班级人数,输入每个人的学号,姓名,和语文分数.数学分数和英语分数(要求使用结构体), //求班级里两个语文分数是最高分的学生的所有信息:数学分数是最高分的两个学生的所有信息:英语平均分 建立 ...
- JavaScript学习记录总结(七)——dom对象应用之用户简单管理
<!DOCTYPE html><html><head><title>users.html</title> <meta name=&qu ...
- 查看PHP的配置信息
查看PHP的配置信息其实一个函数就搞定了. 首先在服务器的根目录建立phpinfo.php文件. 然后打开此文件输入以下内容 <?php phpinfo(); ?> 保存此文件之后在浏览器 ...
- DELPHI WEBSERVICE
一.服务程序 1.依次选择 NEW -> OTHER -> WEB SERVICE -> SOAP SERVER APPLICATION -> ISAPI DYNAMIC LI ...
- poj2367 拓扑序
题意:有一些人他们关系非常复杂,一个人可能有很多后代,现在要制定一种顺序,按照前辈在后代前排列就行 拓扑序裸题,直接建边拓扑排序一下就行了. #include<stdio.h> #incl ...