【C++11特性篇】lambda表达式玩法全解

简介: 【C++11特性篇】lambda表达式玩法全解

一.lambda解决个性化排序问题

[1]设计商品结构体

  • 设计一个商品结构体如下所示
struct Goods
{
    string _name;  // 名字
    double _price; // 价格
    int _evaluate; // 评价
    Goods(const char* str, double price, int evaluate)
        :_name(str)
        , _price(price)
        , _evaluate(evaluate)
    {}
};

[2]利用仿函数(函数对象)解决个性化排序一览

  • 如下所示,利用sort函数+仿函数可以实现不同的个性化排序
  • 价格排大,价格排小,名字排大,名字排小…
//struct ComparePriceLess
struct Compare1
{
    bool operator()(const Goods& gl, const Goods& gr)
    {
        return gl._price < gr._price;
    }
};
//struct ComparePriceGreater
struct Compare2
{
    bool operator()(const Goods& gl, const Goods& gr)
    {
        return gl._price > gr._price;
    }
};
struct CompareEvaluateGreater
{
    bool operator()(const Goods& gl, const Goods& gr)
    {
        return gl._evaluate > gr._evaluate;
    }
};
int main()
{
  vector<Goods> v = { { "苹果", 2.1, 5 }, { "香蕉", 3, 4 }, { "橙子", 2.2, 3 }, { "菠萝", 1.5, 4 } };
  //sort(v.begin(), v.end(), Compare1());  // 价格升序
  //sort(v.begin(), v.end(), Compare2());  // 价格降序
  //sort(v.begin(), v.end(), CompareEvaluateGreater());  // 评价的降序
  sort(v.begin(), v.end(), Compare1()); 
  sort(v.begin(), v.end(), Compare2());
  return 0;
}

[3]利用lambda解决个性化排序一览

为什么要引入lambda?

  • 我们可以观察[2]中用仿函数解决个性化排序会出现一个问题
  • 我们如果看到CompareEvaluateGreater()这个仿函数,我们能知道它是根据"评价的降序"来进行排序
  • 但是当我们看到Compare1()/Compare2(),我们并不能很直观知道它是根据什么来排序,需要找到该函数才明白
  • 以下是改造成lambda形式的基本使用
  • 具体详细的介绍部分在本篇博客的板块二中,这里展示基本使用方法[捕捉列表] (参数列表) mutable -> 返回值类型 { 函数体 }
  • 首先我们要知道,lamda其实是一个局部的匿名函数对象,常与auto搭配使用
//[捕捉列表] (参数列表) mutable -> 返回值类型 { 函数体 }
  // 局部的匿名函数对象
  auto less = [](int x, int y)->bool {return x < y; };
  cout << less(1, 2) << endl;
  • 以下是改造成lambda形式的个性化排序
int main()
{
  vector<Goods> v = { { "苹果", 2.1, 5 }, { "香蕉", 3, 4 }, { "橙子", 2.2, 3 }, { "菠萝", 1.5, 4 } };
  sort(v.begin(), v.end(), Compare1());
  sort(v.begin(), v.end(), Compare2());
  //auto goodsPriceLess = [](const Goods& x, const Goods& y)->bool {return x._price < y._price; };
  //                                                 没有返回值时此部分可省略
  auto goodsPriceLess = [](const Goods& x, const Goods& y){return x._price < y._price; };
  cout << goodsPriceLess(v[0], v[1]) << endl;
  sort(v.begin(), v.end(), goodsPriceLess);
  sort(v.begin(), v.end(), [](const Goods& x, const Goods& y) {
    return x._price < y._price; });
  sort(v.begin(), v.end(), [](const Goods& x, const Goods& y) {
    return x._price > y._price;});
  sort(v.begin(), v.end(), [](const Goods& x, const Goods& y) {
    return x._evaluate < y._evaluate;});
  sort(v.begin(), v.end(), [](const Goods& x, const Goods& y) {
    return x._evaluate > y._evaluate;});
  return 0;
}

二.lambda表达式语法

1)lambda表达式总览

lambda表达式书写格式:[capture-list] (parameters) mutable -> return-type { statement }

  1. lambda表达式各部分说明
    [capture-list] : 捕捉列表,该列表总是出现在lambda函数的开始位置,编译器根据[]来判断接下来的代码是否为lambda函数,捕捉列表能够捕捉上下文中的变量供lambda函数使用。
  • (parameters):参数列表。与普通函数的参数列表一致,如果不需要参数传递,则可以连同()一起省略
  • mutable:默认情况下,lambda函数总是一个const函数,mutable可以取消其常量性。使用该修饰符时,参数列表不可省略(即使参数为空)。
  • ->returntype:返回值类型。用追踪返回类型形式声明函数的返回值类型,没有返回值时此部分可省略。返回值类型明确情况下,也可省略,由编译器对返回类型进行推导。
  • {statement}:函数体。在该函数体内,除了可以使用其参数外,还可以使用所有捕获到的变量。
  • 在lambda函数定义中,参数列表和返回值类型都是可选部分,而捕捉列表和函数体可以为空。因此C++11中最简单的lambda函数为:[]{}; 该lambda函数不能做任何事情。

2) lambda的返回值类型一般可以省略

  • ->returntype:返回值类型。
  • 用追踪返回类型形式声明函数的返回值类型,没有返回值时此部分可省略。返回值类型明确情况下,也可省略,由编译器对返回类型进行推导。
int main()
{
  int a = 0, b = 2;
  double rate = 2.555;
  auto add1 = [](int x, int y)->int {return x + y; };
  auto add2 = [](int x, int y) {return x + y; };//返回值类型明确情况下,也可省略,由编译器对返回类型进行推导。
}

3) 捕捉列表能够捕捉上下文中的变量供lambda函数使用

  • 下面代码需要捕捉rate变量给后面函数体用
auto add3 = [rate](int x, int y) {return (x + y)* rate; };

4) 捕捉列表【特殊使用方式】一览

  • [var]:表示值传递方式捕捉变量var
  • [=]:表示值传递方式捕获所有父作用域中的变量(包括this)
  • [&var]:表示引用传递捕捉变量var
  • [&]:表示引用传递捕捉所有父作用域中的变量(包括this)
  • [this]:表示值传递方式捕捉当前的this指针
  • 以下为 [&] 演示+ [&,a] 演示
int a = 0;
int b = 1;
int c = 2;
int d = 3;
const int e = 1;
cout << &e << endl;
// 引用的方式捕捉所有对象,除了a
// a用传值的方式捕捉
//捕捉所有对象auto func = [&]{函数体};
auto func = [&, a] {
  //a++;fail
  b++;
  c++;
  d++;
  //e++;
  cout << &e << endl;

5) mutable在【传值传参】时的用法

  • 默认情况下,lambda函数总是一个const函数,mutable可以取消其常量性。使用该修饰符时,参数列表不可省略(即使参数为空)
//这里程序是会报错的,可以这样理解:因为其参数默认是带const的,不能被修改
int x = 0, y = 2;
auto swap1 = [add1](int& x, int& y) {
  int tmp = x;
  x = y;
  y = tmp;
  cout << add1(x, y) << endl;
};
swap1(a, b);
int x = 0, y = 2;
auto swap1 = [x, y]() mutable {
  // mutable让捕捉的x和y可以改变了,
  // 但是他们依旧是外面x和y的拷贝
    int tmp = x;
    x = y;
    y = tmp;
  };
  swap1();

6)lambda在实现交换函数swap()时常用【传引用传参】而不是【传值传参+mutable】

  • 在小点4中,我们是用下面代码实现swap()
int x = 0, y = 2;
auto swap1 = [x, y]() mutable {
  // mutable让捕捉的x和y可以改变了,
  // 但是他们依旧是外面x和y的拷贝
    int tmp = x;
    x = y;
    y = tmp;
  };
  swap1();
  • 但是这一回略显繁琐,我们一般用【传引用传参】而不是【传值传参+mutable】,如下所示:
// 引用的方式捕捉
  int x = 0, y = 2;
  auto swap2 = [&x, &y](){
    int tmp = x;
    x = y;
    y = tmp;
  };
  swap2();

三.仿函数(函数对象)与【lambda表达式】的底层其实是一样的

[1]仿函数(函数对象)基本知识回顾

  • 函数对象,又称为仿函数,即可以像函数一样使用的对象,就是在类中重载了operator()运算符的类对象
class Rate
{
public:
    Rate(double rate) : _rate(rate)
    {}
    double operator()(double money, int year)
    {
        return money * _rate * year;
    }
private:
    double _rate;
};
int main()
{
    // 函数对象
    double rate = 0.49;
    Rate r1(rate);
    r1(10000, 2);
    // lambda
    //[=]:表示值传递方式捕获所有父作用域中的变量(包括this)
    auto r2 = [=](double monty, int year)->double {return monty * rate * year;
    };
    r2(10000, 2);
    return 0;
}

[2]如果定义了一个lambda表达式,编译器会自动生成一个类,在该类中重载了operator()

  • 我们通过反汇编可以发现,底层编译器对于lambda表达式的处理方式,完全就是按照函数对象的方式处理的
  • 也就是说:如果定义了一个lambda表达式,编译器会自动生成一个类,在该类中重载了operator()


相关文章
|
2月前
|
编译器 程序员 定位技术
C++ 20新特性之Concepts
在C++ 20之前,我们在编写泛型代码时,模板参数的约束往往通过复杂的SFINAE(Substitution Failure Is Not An Error)策略或繁琐的Traits类来实现。这不仅难以阅读,也非常容易出错,导致很多程序员在提及泛型编程时,总是心有余悸、脊背发凉。 在没有引入Concepts之前,我们只能依靠经验和技巧来解读编译器给出的错误信息,很容易陷入“类型迷路”。这就好比在没有GPS导航的年代,我们依靠复杂的地图和模糊的方向指示去一个陌生的地点,很容易迷路。而Concepts的引入,就像是给C++的模板系统安装了一个GPS导航仪
125 59
|
22天前
|
算法 编译器 C++
【C++11】lambda表达式
C++11 引入了 Lambda 表达式,这是一种定义匿名函数的方式,极大提升了代码的简洁性和可维护性。本文详细介绍了 Lambda 表达式的语法、捕获机制及应用场景,包括在标准算法、排序和事件回调中的使用,以及高级特性如捕获 `this` 指针和可变 Lambda 表达式。通过这些内容,读者可以全面掌握 Lambda 表达式,提升 C++ 编程技能。
49 3
|
22天前
|
安全 编译器 C++
【C++11】新特性
`C++11`是2011年发布的`C++`重要版本,引入了约140个新特性和600个缺陷修复。其中,列表初始化(List Initialization)提供了一种更统一、更灵活和更安全的初始化方式,支持内置类型和满足特定条件的自定义类型。此外,`C++11`还引入了`auto`关键字用于自动类型推导,简化了复杂类型的声明,提高了代码的可读性和可维护性。`decltype`则用于根据表达式推导类型,增强了编译时类型检查的能力,特别适用于模板和泛型编程。
22 2
|
2月前
|
存储 编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(三)
【C++】面向对象编程的三大特性:深入解析多态机制
|
2月前
|
存储 编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(二)
【C++】面向对象编程的三大特性:深入解析多态机制
|
2月前
|
C++
C++ 20新特性之结构化绑定
在C++ 20出现之前,当我们需要访问一个结构体或类的多个成员时,通常使用.或->操作符。对于复杂的数据结构,这种访问方式往往会显得冗长,也难以理解。C++ 20中引入的结构化绑定允许我们直接从一个聚合类型(比如:tuple、struct、class等)中提取出多个成员,并为它们分别命名。这一特性大大简化了对复杂数据结构的访问方式,使代码更加清晰、易读。
41 0
|
19天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
29 2
|
25天前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
61 5
|
1月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
66 4
|
1月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
76 4