草庐IT

C++ Lambda表达式的常见用法

忆_恒心 2023-04-10 原文

⭐️我叫忆_恒心,一名喜欢书写博客的在读研究生👨‍🎓。
如果觉得本文能帮到您,麻烦点个赞👍呗!

近期会不断在专栏里进行更新讲解博客~~~ 有什么问题的小伙伴 欢迎留言提问欧,喜欢的小伙伴给个三连支持一下呗。👍⭐️❤️

前言

作为C++11的新特性的Lambada表达式本身书写倒不是什么难事,网上找一找也能轻易写出正确地表达式,但是在那些场景下使用以及应该如何使用反而会让很多新手困惑。

Lambda表达式

1 表达式定义

构造一个闭包:一个能够捕获作用域中变量的未命名函数对象。

在操作上,闭包是将函数与环境一起存储的记录。

2 书写规范

主要分成六个部分,其中1,2,3,5,6是需要重点掌握的。

  1. capture 子句(在 C++ 规范中也称为 lambda 引入器。)
  2. 参数列表自选。(也称为 lambda 声明符)
  3. 可变规范自选。
  4. 异常规范自选。
  5. 尾随返回类型自选。
  6. λ体。

3 书写表达式

书写[] 用于捕获:

3.1 捕获的基本规则

  • [] - 不捕捉任何变量
  • [&] - 捕获外部作用域中所有变量,并作为引用在函数体内使用 (按引用捕
    获)
  • [=] - 捕获外部作用域中所有变量(依赖于编译器,只能捕获不被优化的变量),并作为副本在函数体内使用 (按值捕获)
    注:拷贝的副本在匿名函数体内部为const类型.
  • [=, &foo] - 按值捕获外部作用域中所有变量,并按照引用捕获外部变量 foo
  • [bar] - 按值捕获 bar 变量,同时不捕获其他变量
  • [&bar] - 按引用捕获 bar 变量,同时不捕获其他变量
  • [this] - 捕获当前类中的 this 指针
  • 让 lambda 表达式拥有和当前类成员函数同样的访问权限
    如果已经使用了 & 或者 =, 默认添加此选项

3.2 使用的场景

用到函数指针的地方,应该考虑一下这个地方需不需要写Lambada表达式。

类中定义了只使用一次的方法,可以考虑是否需要用到Lambada表达式。

只需要使用一次

类中进行调用。

好处:

  • 简化代码: 维护代码完整性
  • 捕获列表 简化代码解耦。

具体的例子:

01、代码维护上

Lambada类似于函数指针的东西,比如sort函数 自定义排序.

auto compare = [](int a, int b)
{
	return a > b;
};

02、代码解耦

如果某天多了一个条件,那么普通的函数参数列表要跟着改变,但是可以使用用匿名函数进行解耦。

例子:

int lambadaParams()
{
    bool cond1 = false;
    bool cond2 = false;
    bool cond3 = true;
    bool cond4 = true;
    auto fun = [=](int t)
    {
        if (cond1)
        {
            cout << "cond1 is true" << endl;
            cout << t << endl;
        }
        if (cond1 && cond2)
        {
            cout << "cond1 && cond2" << endl;
            cout << t << endl;
        }
        if (cond3 || cond4)
        {
            cout << "cond3 || cond4" << endl;
            cout << t << endl;
        }
    };
    if (true)
    {
        fun(1);
    }
    if (true)
    {
        fun(2);
    }
    if (true)
    {
        fun(3);
    }
}

3.3 细节——需要对捕获的值进行修改

由于捕获的值在函数内部是const类型,直接修改会报错。

注:拷贝的副本在匿名函数体内部为const类型.

但是可以添加可变规范

    int num1 = 2;
    int num2 = 3;
    auto fun3 = [=](int a, int b) mutable
    {
        int p = num1 + num2;
        int q = (num1--) + (num2--);
    };
    fun3(1, 2);

3.4 细节——接收返回值时需要注意

使用 auto 类型说明符声明的变量不能出现在其自身的初始值设定项中。

正确修改:

    std::function<int(int)> fun2 = [&fun2](int a)
    {
        if (a == 1)
            return 1;
        if (a == 2)
            return 2;
        else
            return fun2(a - 1) + fun2(a - 2);
    };
    auto x2 = fun2(5);

运行:

3.5 细节——不写返回值会报错的情况

虽然一般Lambada可以不写返回值类型。

当编译器无法进行推断出来类型的时候就会报错。

只需要指明类型就科一了

    auto fun1 = []() -> std::initializer_list<int>
    {
        return {1, 2, 3, 4, 5};
    };
    auto x = fun1();

运行:

还有一种情况需要注意的是当返回值多个类型的时候,根据某个条件进行返回不同的类型这个时候也需要指明返回值类型。

3.6 细节——其他

引用捕获有性能代价么?

需要,这种效率比较细微,体现在

=引用捕获只针对不被编译器优化的部分:

只会捕获使用的变量而不是全部变量,编译器优化问题。不过我用的gcc版本问题对a1进行了初始化,因此fun4依然可以捕获到

例如:

    int a1, b1, c1;
    auto fun4 = [=](int num1, int num2) mutable
    {
        num1++;
        num2++;
        num1 = a1 + num1;
    };
    fun4(0, 1);

完整的实验代码:

#include <iostream>
#include <algorithm>
#include <vector>
#include <functional>
using namespace std;
int main()
{
    vector<int> vec{3, 5, 1, 8, 2, 9, 0};
    auto compare = [](int a, int b)
    {
        return a > b;
    };
    sort(vec.begin(), vec.end(), compare);

    auto fun1 = []() -> std::initializer_list<int>
    {
        return {1, 2, 3, 4, 5};
    };
    auto x = fun1();

    std::function<int(int)> fun2 = [&fun2](int a)
    {
        if (a == 1)
            return 1;
        if (a == 2)
            return 2;
        else
            return fun2(a - 1) + fun2(a - 2);
    };
    auto x2 = fun2(5);

    int num1 = 2;
    int num2 = 3;
    auto fun3 = [=](int a, int b) mutable
    {
        int p = num1 + num2;
        int q = (num1--) + (num2--);
    };
    fun3(1, 2);

    int a1, b1, c1;
    auto fun4 = [=](int num1, int num2) mutable
    {
        num1++;
        num2++;
        num1 = a1 + num1;
    };
    fun4(0, 1);
    return 0;
}

总结

Lambda表达式是C++11重要的特性,使得代码变得更加简洁,易于维护,但是同时也可能导致影响性能,因此使用的时候要注意应用场景。

  1. [capture list] (params list) 关键字-> return type {function body}
  2. captrue list:捕获外部变量列表(选填)
  3. params list:形参列表(可省略)
  4. 关键字:mutable,exception,attribute(可省略)
  5. return type:返回类型(可省略)
  6. function body:函数体

其中:

mutable说明lambda表达式体内的代码可以修改被捕获的变量,并且可以访问被捕获的对象的non-const方法。

exception说明lambda表达式是否抛出异常以及何种异常。

attribute用来声明属性。

特点:

​ 一次性使用

作用:

  • 总体上:代码完整性
  • 局部:闭包是的传参数更容易

最后,最后
如果觉得有用,麻烦三连👍⭐️❤️支持一下呀,希望这篇文章可以帮到你,你的点赞是我持续更新的动力

有关C++ Lambda表达式的常见用法的更多相关文章

  1. ruby 正则表达式 - 如何替换字符串中匹配项的第 n 个实例 - 2

    在我的应用程序中,我需要能够找到所有数字子字符串,然后扫描每个子字符串,找到第一个匹配范围(例如5到15之间)的子字符串,并将该实例替换为另一个字符串“X”。我的测试字符串s="1foo100bar10gee1"我的初始模式是1个或多个数字的任何字符串,例如,re=Regexp.new(/\d+/)matches=s.scan(re)给出["1","100","10","1"]如果我想用“X”替换第N个匹配项,并且只替换第N个匹配项,我该怎么做?例如,如果我想替换第三个匹配项“10”(匹配项[2]),我不能只说s[matches[2]]="X"因为它做了两次替换“1fooX0barXg

  2. ruby - 正则表达式将非英文字母匹配为非单词字符 - 2

    @raw_array[i]=~/[\W]/非常简单的正则表达式。当我用一些非拉丁字母(具体来说是俄语)尝试时,条件是错误的。我能用它做什么? 最佳答案 @raw_array[i]=~/[\p{L}]/使用西里尔字符进行测试。引用:http://www.regular-expressions.info/unicode.html#prop 关于ruby-正则表达式将非英文字母匹配为非单词字符,我们在StackOverflow上找到一个类似的问题: https://

  3. git使用常见问题(提交代码,合并冲突) - 2

    文章目录git常用命令(简介,详细参数往下看)Git提交代码步骤gitpullgitstatusgitaddgitcommitgitpushgit代码冲突合并问题方法一:放弃本地代码方法二:合并代码常用命令以及详细参数gitadd将文件添加到仓库:gitdiff比较文件异同gitlog查看历史记录gitreset代码回滚版本库相关操作远程仓库相关操作分支相关操作创建分支查看分支:gitbranch合并分支:gitmerge删除分支:gitbranch-ddev查看分支合并图:gitlog–graph–pretty=oneline–abbrev-commit撤消某次提交git用户名密码相关配置g

  4. ruby - 正则表达式在哪个位置失败? - 2

    我需要一个非常简单的字符串验证器来显示第一个符号与所需格式不对应的位置。我想使用正则表达式,但在这种情况下,我必须找到与表达式相对应的字符串停止的位置,但我找不到可以做到这一点的方法。(这一定是一种相当简单的方法……也许没有?)例如,如果我有正则表达式:/^Q+E+R+$/带字符串:"QQQQEEE2ER"期望的结果应该是7 最佳答案 一个想法:你可以做的是标记你的模式并用可选的嵌套捕获组编写它:^(Q+(E+(R+($)?)?)?)?然后你只需要计算你获得的捕获组的数量就可以知道正则表达式引擎在模式中停止的位置,你可以确定匹配结束

  5. ruby - 有没有办法从 ruby​​ case 语句中访问表达式? - 2

    我想从then子句中访问c​​ase语句表达式,即food="cheese"casefoodwhen"dip"then"carrotsticks"when"cheese"then"#{expr}crackers"else"mayo"end在这种情况下,expr是食物的当前值(value)。在这种情况下,我知道,我可以简单地访问变量food,但是在某些情况下,该值可能无法再访问(array.shift等)。除了将expr移出到局部变量然后访问它之外,是否有直接访问caseexpr值的方法?罗亚附注我知道这个具体示例很简单,只是一个示例场景。 最佳答案

  6. ruby - 正则表达式 - 排除一个字符 - 2

    这是一个例子:s="abcd+subtext@example.com"s.match(/+[^@]*/)Result=>"+subtext"问题是,我不想在其中包含“+”。我希望结果是“潜台词”,没有+ 最佳答案 您可以在正则表达式中使用括号来创建匹配组:s="abcd+subtext@example.com"s=~/\+([^@]*)/&&$1=>"subtext" 关于ruby-正则表达式-排除一个字符,我们在StackOverflow上找到一个类似的问题:

  7. ruby - 如何遍历 Ruby 中所有正则表达式匹配的字符串? - 2

    我们有一个字符串:“”这个正则表达式://i如何从当前字符串中获取所有匹配项? 最佳答案 "".scan(//)参见scan在ruby​​-docs上 关于ruby-如何遍历Ruby中所有正则表达式匹配的字符串?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/6857852/

  8. ruby - 将对象设置为 nil 是否很常见? - 2

    我正在构建一个应用程序,想知道是否将未使用的对象设置为nil是生产级编码中的常见做法。我知道这只是垃圾收集器的提示,并不总是处理对象。 最佳答案 根据这个thread如果您使用完一个成员对象,将其设置为nil将引发被引用对象被垃圾回收。如果它是局部变量,方法exit将做同样的事情。也就是说,如果您要求将成员显式设置为nil,我会质疑您的设计。 关于ruby-将对象设置为nil是否很常见?,我们在StackOverflow上找到一个类似的问题: https://

  9. Ruby 正则表达式匹配逗号,但忽略括号中的逗号 - 2

    我正在尝试通过正则表达式拆分参数列表。这是一个带有我的参数列表的字符串:"a=b,c=3,d=[1,3,5,7],e,f=g"我想要的是:["a=b","c=3","d=[1,3,5,7]","e","f=g"]我试过先行,但Ruby不允许使用动态范围后行,所以这行不通:/(?如何让正则表达式忽略方括号中的所有内容? 最佳答案 也许这样的东西对你有用:str.scan(/(?:\[.*?\]|[^,])+/)编辑再三考虑。简单的非贪婪匹配器在某些嵌套括号的情况下会失败。 关于Ruby正则

  10. ruby - 查找重叠的正则表达式匹配项 - 2

    我想找到给定字符串中的所有匹配项,包括重叠匹配项。我怎样才能实现它?#Example"a-b-c-d".???(/\w-\w/)#=>["a-b","b-c","c-d"]expected#Solutionwithoutoverlappedresults"a-b-c-d".scan(/\w-\w/)#=>["a-b","c-d"],but"b-c"ismissing 最佳答案 在积极的前瞻中使用捕获:"a-b-c-d".scan(/(?=(\w-\w))/).flatten#=>["a-b","b-c","c-d"]参见Rubyde

随机推荐