lambda表达式的求值-对象构造

本来想写“定义”,即“definition”,像函数定义一样,函数具体实现的代码实体即为实现,但是就像lambda既然被称为表达式,它确实有表达式那样“求值”的动作,而不仅仅像函数那样静态地编译。所以应该写“求值”更确切些,即“evaluation”。如果lambda定义的仅仅是一个函数,也就是返回一个函数指针,那么这里就应该叫做“定义”,但是lambda实际上定义了一个“函数对象”,即“function object”。

首先像下面这样定义一个简单的lambda表达式:

设好断点,在调试状态下查看相应的汇编代码:

这是设置为不显示符号名称后更原汁原味的汇编代码:

可以看到,编译器将a,b的地址压栈,这里都是地址,虽然对于变量a我们是by value而不是by reference。接下来,把lbd的地址装入ecx,然后call,很明显的thiscall,那么lbd自然就是一个object,那么此时此刻被调用的当然就是constructor。

跟踪到constructor代码处继续看,跳过函数入口的stack frame、寄存器保存等操作,直接定位到目标代码:

不显示符号名称,对比着看:

首先,把ecx的值存到堆栈上的this指针,然后通过堆栈上a的地址取到a的值,把a的值存储到this所指对象(即上面的lbd)0偏移处,占4字节(一个int),最后把堆栈上b的地址存储到this所指对象4字节偏移处,占4字节(32位平台指针大小)。一个值、一个地址,在这里我们就可以体会到“by value”和“by reference”的实现原理了。a的值和b的地址在这里就被存储到了lambda对象体内部,对程序员不可见,在对象的生命周期内都不会改变。

所以我们可以推导出,lbd对象的class定义为:

class Clambda
{
public:
    Clambda(int &a, int &b)
    {
        this->a = a;
        this->pb = &b;
    }
protected:
    int a;
    int *pb;
};

或者写成这样:

class Clambda
{
public:
    Clambda(int &a, int &b) : a(a), b(b)
    {
    }
protected:
    int a;
    int &b;
};

lambda表达式的定义,即表达式的求值,也就是等价类对象的实例化。

lambda表达式的Function Call Operator

在原有代码基础之上,加一句调用代码:

在调试汇编窗口查看:

不显示符号名称:

不出所料,在这里我们又看到了thiscall。继续跟踪到函数内部:

不显示符号名称:

首先从栈上取到x的值,然后通过this指针取到对象体0偏移处4个字节即变量a的值(“by value”,这个值是lambda对象定义时刻的值),然后相乘,之后通过this指针取到对象体4字节偏移处存储的指向变量b的指针,进一步得到b的值(“by reference”,当前时刻的值),再相乘得到结果。

现在,我们可以完善lbd的class定义:

class Clambda
{
public:
    Clambda(int &a, int &b)
    {
        this->a = a;
        this->pb = &b;
    }
    int operator()(int x)
    {
        return x * a * *pb;
    }
protected:
    int a;
    int *pb;
};

或者:

class Clambda
{
public:
    Clambda(int &a, int &b) : a(a), b(b)
    {
    }
    int operator()(int x)
    {
        return x * a * b;
    }
protected:
    int a;
    int &b;
};

用class实现等价lambda

把原来代码中的lambda表达式用我们上面定义的Clambda类替换,汇编代码如下:

对象构造和function call:

不显示符号名称:

构造函数:

不显示符号名称:

function call函数体:

不显示符号名称:

结束语

lambda的出现大大方便了function object的创建,编译器帮我们做了多余的工作。然而lambda带来方便的同时,也有很多值得注意的地方,例如笔者在做Win8 Metro开发的时候,就曾不止一次内存访问异常,究其原因就是因为异步lambda按引用捕获的外层函数局部变量已销毁。了解了lambda的底层原理,应该可以帮我们更好的避免此类错误,用好lambda。

喜爱C++,喜爱lambda。

C++对象模型之lambda表达式的更多相关文章

  1. 十二、C# 委托与Lambda表达式(匿名方法的另一种写法)

    委托与Lambda表达式   1.委托概述 2.匿名方法 3.语句Lambda 4.表达式Lambda 5.表达式树   一.委托概述 相当于C++当中的方法指针,在C#中使用delegate 委托来 ...

  2. 你知道C#中的Lambda表达式的演化过程吗?

    那得从很久很久以前说起了,记得那个时候... 懵懂的记得从前有个叫委托的东西是那么的高深难懂. 委托的使用 例一: 什么是委托? 个人理解:用来传递方法的类型.(用来传递数字的类型有int.float ...

  3. Linq表达式、Lambda表达式你更喜欢哪个?

    什么是Linq表达式?什么是Lambda表达式? 如图: 由此可见Linq表达式和Lambda表达式并没有什么可比性. 那与Lambda表达式相关的整条语句称作什么呢?在微软并没有给出官方的命名,在& ...

  4. 背后的故事之 - 快乐的Lambda表达式(一)

    快乐的Lambda表达式(二) 自从Lambda随.NET Framework3.5出现在.NET开发者眼前以来,它已经给我们带来了太多的欣喜.它优雅,对开发者更友好,能提高开发效率,天啊!它还有可能 ...

  5. Kotlin的Lambda表达式以及它们怎样简化Android开发(KAD 07)

    作者:Antonio Leiva 时间:Jan 5, 2017 原文链接:https://antonioleiva.com/lambdas-kotlin/ 由于Lambda表达式允许更简单的方式建模式 ...

  6. java8中lambda表达式的应用,以及一些泛型相关

    语法部分就不写了,我们直接抛出一个实际问题,看看java8的这些新特性究竟能给我们带来哪些便利 顺带用到一些泛型编程,一切都是为了简化代码 场景: 一个数据类,用于记录职工信息 public clas ...

  7. 背后的故事之 - 快乐的Lambda表达式(二)

    快乐的Lambda表达式 上一篇 背后的故事之 - 快乐的Lambda表达式(一)我们由浅入深的分析了一下Lambda表达式.知道了它和委托以及普通方法的区别,并且通过测试对比他们之间的性能,然后我们 ...

  8. CRL快速开发框架系列教程二(基于Lambda表达式查询)

    本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...

  9. Lambda 表达式递归用法实例

    注意: 使用Lambda表达式会增加额外开销,但却有时候又蛮方便的. Windows下查找子孙窗口实例: HWND FindDescendantWindows(HWND hWndParent, LPC ...

随机推荐

  1. jQuery和AngularJS的区别

    这篇文章主要介绍了jQuery和AngularJS的区别浅析,本文着重讲解一个熟悉jQuery开的程序员如何应对AngularJS中的一些编程思想的转变,需要的朋友可以参考下   最近一直在研究ang ...

  2. Uva11582

    最近各种破事忙死了 终于开始做题了 紫薯第10章第一题,come on 设g(i)=f(i) mod n,当二元组(g(i).g(i+1))出现重复时,整个序列就开始重复(这一话怎么也不懂,请大神解释 ...

  3. Noip2016组合数(数论)

    题目描述 组合数表示的是从n个物品中选出m个物品的方案数.举个例子,从(1,2,3) 三个物品中选择两个物品可以有(1,2),(1,3),(2,3)这三种选择方法.根据组合数的定 义,我们可以给出计算 ...

  4. [POJ 1410] Intersection(线段与矩形交)

    题目链接:http://poj.org/problem?id=1410 Intersection Time Limit: 1000MS   Memory Limit: 10000K Total Sub ...

  5. OOAD-设计模式(三)之创建型设计模式(5种)

    前言 前面介绍了OOAD的基础知识,现在我们来详细的说明一下GOF设计模式中的23种模式,希望大家能够学到东西! 一.工厂方法模式(Factory Method) 1.1.工厂方法模式概述 工厂方法模 ...

  6. sql执行报错--This version of MySQL doesn't yet support 'LIMIT & IN/ALL/ANY/SOME subquery'

    问题: 不支持使用 LIMIT 子句的 IN/ALL/ANY/SOME 子查询,即是支持非 IN/ALL/ANY/SOME 子查询的 LIMIT 子查询. 解决: 将语句:select * from ...

  7. position,display,float,overflow,margin,padding之间的相互影响

    1.元素分为块级元素和行内元素, 块级元素可以设置宽高,会自动换行,并且会发生相邻margin的合并问题.行内元素设置宽和高无效,以水平方向排列,(行内元素,绝对定位,浮动元素不会发生外边距合并)并且 ...

  8. 关于php的命名空间

    php定义命名空间要使用namespace关键字,例:namespace Database 使用命名空间中的类要使用use关键字,也可以在use后面加as给类取别名,例:use Database\SQ ...

  9. 使用Mongodb+Shiro+SpringMVC实现动态权限分配

    此次的文档只对Mongodb整合Shiro并且实现动态权限分配做整理,其它的内容以后会补上. 第一步.创建在web.xml中配置 Spring .Shiro shiroFilter 过滤器是用来将请求 ...

  10. 如何成为一个javascript高手【转载】

      原文网址: http://www.cnblogs.com/keva/p/how-to-become-a-javascript-badass.html 英文网址:http://www.clientc ...