[Deepseek 学c++]初始化捕获与按值(显式与隐式)捕获傻傻分不清

之前effective Modern C++ 条款31 看了几次似懂非懂,现在终于理解了这一章。

1. 静态变量可以在lambda 表达式使用,但不能被捕获

考虑以下这道题:
在这里插入图片描述
初始化捕获和默认值捕获的结果 a 值是什么? deepseek 第二个给的答案是错的。
在这里插入图片描述
注意静态变量,与引用传递和值传递无关, 只能捕获非静态局部变量。effective 里说的
以上说法不对。

        static int y = 20;
        auto f = [y = y]() mutable { ++y; };
        f(); 
        return y; 输出20

        static int y = 20;
        auto f = [=]() mutable { ++y; };
        f(); 
        return y; y 为21
不捕获
        static int y = 20;
        auto f = []() mutable { ++y; };
        f(); 
        return y;  输出21

        static int y = 20;
        auto f = [&]() mutable { ++y; };
        f(); 
        return y; 21

        static int y = 20;
        auto f = [&y = y]() mutable { ++y; };
        f(); 
        return y; 21

[&a = a]: by-reference capture with an initializer
相当于在lambda函数体中加了一个新的变量auto &a, 其初值为函数作用域内的 static int a;引用,对新的a操作实际还是修改的是static int a
[a = a]:by-copy capture with an initializer
相当于 lambda函数体中加了一个新的变量 auto a, 其初值为外部的 static int a;操作是新的a

        int y = 20;
        auto f = [y = y]() mutable { ++y; };
        f(); 
        return y; 20
        
   int y = 20;
        auto f = [=]() mutable { ++y; };
        f(); 
        return y; 20

闭包与闭包类

闭包通常指的是函数对象,也就是lambda表达式生成的对象,而闭包类可能是指实现闭包的底层类结构。不过,用户可能对这两个术语的具体区别不太清楚,需要详细解释。

首先,回忆一下C++中的lambda表达式。当写一个lambda时,编译器会生成一个匿名的函数对象类,这个类通常被称为闭包类。而闭包则是这个类的实例,也就是具体的对象。因此,闭包类是类型,而闭包是该类型的对象。用户可能混淆了这两个概念,需要指出它们的层次不同。

接下来,需要详细说明闭包类的结构。比如,lambda捕获的变量会成为闭包类的成员变量,operator()的重载实现函数调用。而闭包则是这个类的一个实例,包含了具体的捕获值。例如,当使用不同的捕获方式(如值捕获、引用捕获)时,闭包类的成员变量类型和存储方式也会不同
闭包类是类型,闭包是实例;**闭包类由编译器生成,包含数据成员和operator(),而闭包是该类的具体对象,持有捕获的变量。**确保用户能够清晰区分这两个概念,并在实际编程中正确使用lambda和理解其底层机制。

2. 为什么需要初始化捕获

对类成员变量的 捕获
输出y 是20, 为什么?

class Solution {
private:
    // int y = 20;
public:
    int y = 20;
    int findMaxForm(vector<string>& strs, int m, int n) {
        auto f = [=]() mutable { ++y; };
        f(); 
        cout << "y:" << y << endl;
        return 0;
    }
}; 输出y 是21, 为什么?

当lambda在类成员函数内部捕获成员变量时,直接使用成员变量名可能不会像普通局部变量那样被捕获。因为成员变量实际上是通过this指针访问的,所以当使用[=]捕获时,实际上会隐式捕获this指针,而不是直接捕获成员变量y。也就是说,lambda中的y实际上是通过this指针访问的类成员变量,而不是一个副本。

而输出y 是20, 为什么?

class Solution {
public:
    int findMaxForm(vector<string>& strs, int m, int n) {
        int y = 20;
        auto f = [=]() mutable { ++y; };
        f(); 
        cout << "y:" << y << endl;
        return 0;
    }
}; 

解释,第一种相当于隐式捕获this指针, 即代码等价于

在这里插入图片描述

什么都不加报错

在这里插入图片描述
在这里插入图片描述

auto f = y = y mutable { ++y; }; 与 [y] 的区别

前者正常通过编译,。
在这里插入图片描述

effective 提到, 按引用或值捕获的变量必须 是lambda 表达式作用域可见的 非静态局部变量, 不能是成员变量。(初始化捕获,即广义捕获除外)
在这里插入图片描述
在这里插入图片描述

初始化捕获、按值捕获,区别

在这里插入图片描述
在这里插入图片描述在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值