c++11 lambda表达式底层

1、概述

lambda表达式也叫匿名函数或闭包,如果这个函数只使用不频繁,或者是不想起名,可以使用匿名函数。
lambda式可以临时制作出回调函数、接口适配函数或语境相关函数的特化版本以供一次性使用。

// 闭包可以复制
auto fun = [capture_list](parameter_list)->return type{fucntion body};

2、捕获方式

捕获是针对于在创建lambda式的作用域内可见的非静态 局部变量(包括形参),捕获的变量放在中括号内。

C++11中有两种捕获模式:按值或按引用, 根据不同的捕获方式,闭包类会持有数据的副本或引用。

void function()
{
	int index = 0;
	std::string str = "abc";
	Widget *pWdt = new Widget();
	
	auto f = [&](){
		pWdt->setName("name"); // 按引用捕获可能导致空悬引用(在function() 结束之后调用该闭包)
	};
}
[ ]:	不能访问任何局部变量
[str]:	按值访问局部变量str(可以使用mutable去掉常量属性)
[&str]:按引用访问局部变量str
[=]:	按值访问所有局部变量(按值默认捕获,一般要避免)
[&]:	按引用访问所有局部变量(按引用默认捕获,一般要避免)
[=,&str]:按引用访问str,按值访问其它局部变量
[&,str]: 按值访问str,按引用访问其它局部变量

注意问题:

  • 一般来说,最好是显式地列出lambda式所依赖的局部变量,避免使用默认捕获模式。
  • 引用捕获局部变量可能导致空悬引用,这时可以改用按值捕获,这样闭包会持有这个指针的副本。
  • 捕获this指针也可能会导致有悬空的指针,这时可以将你想要的成员变量复制到局部变量中,然后捕获该局部副本。或者是使用c++14中的广义lambda捕获。

3、底层原理

从下图可以看出,lambda表达式其实就是一个类模版,在使用时会被实例化。
每个lambda式都会触发编译器生成一个独一无二的闭包类,闭包中的语句变成该类成员函数operator( )中的执行指令。

  • 按引用捕获,闭包类持有数据的引用
    在这里插入图片描述
  • 按值捕获,闭包类持有数据副本
    在这里插入图片描述

不太理解的可以去看看 * 和* &的区别,然后使用这个在线编译的网址体验一下

4、广义lambda捕获(C++14)

广义lambda捕获也叫 初始化捕获,一般使用场景有两个:

  • 需要把一个只移对象放入闭包(如:std::unique_ptr或std::future型别的对象)
  • 想要把一个 复制成本很高,但移动成本很低 的对象放入闭包

c++14为对象移动到闭包提供了直接支持。

class Widget {
public:
	void setTitle(const std::string&);
	void setIcon(const icon&);
	void show();
	
private:
	......
};

auto pWdt = std::make_unique<Widget>();
pWdt->setTitle("title");

auto fun = [pw = std::move(pWdt)]{
	pw->show();
};

// 也可
auto fun1 = [pw = std::make_unique<Widget>()]{
	pw->show();
}

[pw = std::move(pWdt)]
上面这段代码就是初始化捕获,在=左边的,是你指定的闭包类成员变量的名字,右边是这个成员变量的初始化表达式。

如果编译器不支持c++14,也可以使用std::bind模拟初始化捕获。

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
C++ Lambda表达式是一种简洁的匿名表示方法,可以在代码中定义并使用。Lambda表达式的格式一般为:[] (参数列表) mutable -> 返回值型 { 表达式 }。Lambda表达式可以捕获外部变量,并将其作为参数传递给函数体部分进行处理。Lambda表达式在使用时可以作为函数对象、函数指针或者函数参数进行传递。 Lambda表达式底层原理是通过生成一个匿名来实现。该会重载函数调用运算符(),并包含Lambda表达式的函数体。Lambda表达式中捕获的外部变量会以成员变量的形式存储在该中。当Lambda表达式被调用时,实际上是调用了该的重载函数调用运算符()。 Lambda表达式可以与std::function结合使用,以实现函数对象的灵活使用。也可以将Lambda表达式赋值给相同型的函数指针,实现函数指针的使用。但一般不建议这样使用,因为Lambda表达式已经提供了更加方便和简洁的方式。 总结来说,C++ Lambda表达式是一种用于定义匿名函数的语法,可以捕获外部变量并进行处理。其底层通过生成一个匿名来实现,并提供了与std::function和函数指针的结合使用方式。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C++11lambda表达式](https://blog.csdn.net/zhang_si_hang/article/details/127117260)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值