c++,从汇编角度看lambda

本篇作为c++,从汇编底层角度深入理解带捕获的lambda如何转化为std::function的开胃小菜

复制代码
#include <iostream>
#include <functional>

int main(int args, char* argv[]) {
    [](){
        std::cout << "lambda 0" << std::endl;
    }();

    []() -> int {
        std::cout << "lambda 1" << std::endl;
        return 0;
    }();

    [](int a = 1) -> int {
        std::cout << "lambda 1" << std::endl;
        return a;
    }();

    int a = 1, b = 2;

    [a, b]() -> int {
        std::cout << b << std::endl;
        return a;
    }();
}

编译后看一下符号,lambda相关的只有4个,对应main函数中的4个lambda表达式

复制代码
# g++ -std=c++20 -fno-elide-constructors  -O0 -g -o main main.cpp
# readelf -s -W main |c++filt |grep lambda
    13: 0000000000401176    43 FUNC    LOCAL  DEFAULT   14 main::{lambda()#1}::operator()() const
    14: 00000000004011a2    47 FUNC    LOCAL  DEFAULT   14 main::{lambda()#2}::operator()() const
    15: 00000000004011d2    48 FUNC    LOCAL  DEFAULT   14 main::{lambda(int)#3}::operator()(int) const
    16: 0000000000401202    48 FUNC    LOCAL  DEFAULT   14 main::{lambda()#4}::operator()() const

看一下main如何执行这些lambda表达式的:

复制代码
0000000000401232 <main>:
int main(int args, char* argv[]) {

  # 申请了 0x20 大小的栈空间
  401236:	48 83 ec 20          	sub    $0x20,%rsp

  # lea 是加载地址,调用表达式时`rdi`寄存器传入了一个地址
  401241:	48 8d 45 f5          	lea    -0xb(%rbp),%rax
  401245:	48 89 c7             	mov    %rax,%rdi
  401248:	e8 29 ff ff ff       	call   401176 <main::{lambda()#1}::operator()() const>
    }();

  # 看这里和上面的lea加载地址,0xb和0xa,只差1,很奇怪
  40124d:	48 8d 45 f6          	lea    -0xa(%rbp),%rax
  401251:	48 89 c7             	mov    %rax,%rdi
  401254:	e8 49 ff ff ff       	call   4011a2 <main::{lambda()#2}::operator()() const>
    }();
    
  # 第三个 lambda,多传入了 立即数1 作为第二个参数
  401259:	48 8d 45 f7          	lea    -0x9(%rbp),%rax
  40125d:	be 01 00 00 00       	mov    $0x1,%esi
  401262:	48 89 c7             	mov    %rax,%rdi
  401265:	e8 68 ff ff ff       	call   4011d2 <main::{lambda(int)#3}::operator()(int) const>

  # 第四个,出现差异了
  int a = 1, b = 2;
  40127e:	c7 45 fc 01 00 00 00 	movl   $0x1,-0x4(%rbp)
  401285:	c7 45 f8 02 00 00 00 	movl   $0x2,-0x8(%rbp)

  # rdi 是 -0x10(%rbp) 地址,这个地址是(a和b共同空间)的地址,只传递了一个地址!
    [a, b]() -> int {
  40128c:	8b 45 fc             	mov    -0x4(%rbp),%eax
  40128f:	89 45 f0             	mov    %eax,-0x10(%rbp)
  401292:	8b 45 f8             	mov    -0x8(%rbp),%eax
  401295:	89 45 f4             	mov    %eax,-0xc(%rbp)
    }();
  401298:	48 8d 45 f0          	lea    -0x10(%rbp),%rax
  40129c:	48 89 c7             	mov    %rax,%rdi
  40129f:	e8 6e ff ff ff       	call   401212 <main::{lambda()#4}::operator()() const>

总结:

1、写一个 lambda,就会生成一个对应的函数

2、lambda 的传参,和常规函数传参一致,调用时候传递参数

3、不带捕获的 lambda,第一个参数是一个 无意义的 地址值,类似于类的this,但大小是1,无意义

4、带捕获的lambda,第一个参数是参数列表地址(所有参数排序作为整个空间,传递空间地址,就像传递结构体指针)

相关推荐
三体世界3 小时前
Mysql基本使用语句(一)
linux·开发语言·数据库·c++·sql·mysql·主键
John_ToDebug5 小时前
JS 与 C++ 双向通信实战:基于 WebHostViewListener 的消息处理机制
前端·c++·chrome
papership5 小时前
【入门级-C++程序设计:11、指针与引用-引 用】
c语言·开发语言·c++·青少年编程
岁忧5 小时前
(LeetCode 每日一题) 1780. 判断一个数字是否可以表示成三的幂的和 (数学、三进制数)
java·c++·算法·leetcode·职场和发展·go
rhythmcc8 小时前
【visual studio】visual studio配置环境opencv和onnxruntime
c++·人工智能·opencv
十五年专注C++开发9 小时前
CMake进阶: externalproject_add用于在构建阶段下载、配置、构建和安装外部项目
linux·c++·windows·cmake·自动化构建
白书宇10 小时前
5.从零开始写LINUX内核--从实模式到保护模式的过渡实现
linux·汇编·数据库·开源
码与农11 小时前
C++设计模式
开发语言·c++
Tipriest_11 小时前
CMake message()使用指南
c++·cmake·message
企鹅chi月饼12 小时前
c++中的Lambda表达式详解
c++·lambda