当前位置: 首页 > news >正文

C++11 lambda表达式和包装器

C++11 lambda表达式和包装器

  • 一.lambda表达式
    • 1.lambda表达式的引入
    • 2.基本语法和使用
      • 1.基本语法
      • 2.使用
        • 1.传值捕捉的错误之处
        • 2.传引用捕捉
    • 3.lambda表达式的底层原理
    • 4.lambda的特殊之处
    • 5.lambda配合decltype的新玩法
  • 二.function包装器
    • 1.概念
    • 2.包装函数
      • 1.包装普通函数
      • 2.包装成员函数
    • 3.包装器的另一个应用:统一模板实例化的类型
    • 4.小小总结
  • 三.bind包装器
    • 1.概念
    • 2.bind包装器绑定固定参数
      • 1.无意义绑定
      • 2.绑定固定参数
    • 3.bind包装器调整传参顺序

C++11引入的语法当中
除了STL库当中的区别,还有几个很常用的语法
就是我们今天要介绍的lambda表达式和function包装器
(bind包装器不是特别常用,但是我们有些时候也可能会用到,因此我们也介绍一下)

一.lambda表达式

1.lambda表达式的引入

我们之前学习过函数指针和仿函数,它们都是一种回调函数
只不过仿函数的语法更加简单明了

但是仿函数有一个缺点:
实现它必须要定义一个类出来
例如:
我们要实现一个比较算法,对不同的商品按照价格分别升序和降序排序

#include <iostream>
using namespace std;
#include <algorithm>//sort的头文件
#include <vector>
struct Goods
{string _name;//名称double _price;//价格int _evaluate;//评价Goods(const char* str, double price, int evaluate):_name(str), _price(price), _evaluate(evaluate){}
};//按照价格升序
struct ComparePriceLess
{bool operator()(const Goods& left, const Goods& right){return left._price < right._price;}
};//按照价格降序
struct ComparePriceGreater
{bool operator()(const Goods& left, const Goods& right){return left._price > right._price;}
};int main()
{vector<Goods> v = { { "苹果", 2.5, 5 }, { "香蕉", 3.2, 4 }, { "橙子", 2.9, 3 }, { "菠萝", 1.5, 4 } };sort(v.begin(),v.end(), ComparePriceLess());for (auto& e : v){cout << "商品名称: " << e._name << " 价格 :" << e._price << " 评价:" << e._evaluate << endl;}cout << endl;sort(v.begin(), v.end(), ComparePriceGreater());for (auto& e : v){cout << "商品名称: " << e._name << " 价格 :" << e._price << " 评价:" << e._evaluate << endl;}cout << endl;return 0;
}

在这里插入图片描述
我们发现,仅仅是需要分别按照价格升序和降序排序就需要单独写两个类出来
那么如果我又有需求了:分别按照评分升序和降序等等…
那样的话还要再写几个类
而且仿函数的命名上面也有问题,
刚才我们命名的是ComparePriceLess和ComparePriceGreater
如果有人这么命名呢?
Compare1,Compare2…
那样的话代码可读性就很差了

因此C++11引入了lambda表达式来解决这一问题
我们先来用lambda表达式取代一下刚才的仿函数

int main()
{vector<Goods> v = { { "苹果", 2.5, 5 }, { "香蕉", 3.2, 4 }, { "橙子", 2.9, 3 }, { "菠萝", 1.5, 4 } };//按照价格升序sort(v.begin(), v.end(), [](const Goods& left, const Goods& right){return left._price < right._price;});for (auto& e : v){cout << "商品名称: " << e._name << " 价格 :" << e._price << " 评价:" << e._evaluate << endl;}cout << endl;//按照价格降序sort(v.begin(), v.end(), [](const Goods& left, const Goods& right){return left._price > right._price;});for (auto& e : v){cout << "商品名称: " << e._name << " 价格 :" << e._price << " 评价:" << e._evaluate << endl;}cout << endl;return 0;
}

在这里插入图片描述
此时我们每次调用sort的时候只需要传入一个lambda表达式
即可指明比较方式,大大简化代码并且提高了代码的可读性

2.基本语法和使用

1.基本语法

//按照价格升序
sort(v.begin(), v.end(), [](const Goods& left, const Goods& right)
{return left._price < right._price;
});

根据这个排序我们可以看出

lambda表达式的底层就是一个匿名函数
该函数无法直接调用,如果想要直接调用,可借助auto将其赋值给一个对象,用那个对象来调用

在这里插入图片描述
在这里插入图片描述小例子:
在这里插入图片描述
lambda表达式中的捕捉列表可以捕捉上下文中的变量
被捕捉到的变量可以被lambda表达式使用,而且我们可以设置捕捉的方式是传值还是传引用
在这里插入图片描述

2.使用

下面我们用lambda表达式实现一下int类型的swap函数

1.传值捕捉的错误之处
int main()
{int a = 1, b = 2;//这里传值捕捉了a和b,所以后面调用myswap时无需传参auto myswap = [a, b](){int tmp = a;a = b;b = tmp;};myswap();cout << "交换后 : " << " a: " << a <<" b: " << b << endl;return 0;
}

在这里插入图片描述
在这里插入图片描述
所以我们要加上mutable
在这里插入图片描述
可是交换之后a还是1,b还是2啊?
为什么?
因为我们是传值捕捉,形参的改变不会影响实参
因此mutable的意义就是提醒我们
1.如果你想要改变实参,请你传引用捕捉,传值捕捉的话改变的是形参不是实参
2.如果你不想改变实参,传值捕捉的话,需要加上mutable提醒别人这里是传值捕捉,不会影响实参

2.传引用捕捉

方案1:传值捕获a和b,参数列表无需传参
在这里插入图片描述
方案2:捕获列表不去捕获,参数列表传引用传参
在这里插入图片描述

3.lambda表达式的底层原理

其实lambda表达式的底层原理就是仿函数,对operator()进行了重载
就像是范围for的底层原理就是迭代器一样

struct Myswap
{void operator()(int& a, int& b){int tmp = a;a = b;b = tmp;}
};
int main()
{int a = 1, b = 2;//这里捕获列表不去捕获,参数列表需要传参auto myswap = [](int& a,int& b){int tmp = a;a = b;b = tmp;};int x = 9;//为了方便调试打断点看反汇编Myswap myswap2;myswap2(a, b);myswap(a,b);return 0;
}

在这里插入图片描述
在这里插入图片描述
这是因为每个lambda表达式的类型是不同的
(在VS下,lambda会被处理为函数对象,该函数对象对应的类名叫做<lambda_uuid>)
在这里插入图片描述
关于uuid
大家感兴趣的话可以百度一下
UUID
在这里插入图片描述

4.lambda的特殊之处

lambda不支持直接调用默认构造来构造对象
但是支持拷贝构造对象
在这里插入图片描述
在这里插入图片描述

5.lambda配合decltype的新玩法

sort是传入函数对象即可,可是对于priority_queue这种需要传入类的容器来说,lambda就没有用武之地了吗?
并不是的
在这里插入图片描述

//建一个小堆(注意:优先级队列:默认是大堆配less,小堆配greater)
auto f = [](int a, int b) {return a > b; };priority_queue<int, vector<int>, decltype(f)> minheap(f);

在这里插入图片描述

二.function包装器

1.概念

在这里插入图片描述

2.包装函数

下面我们来使用一下function包装器
头文件是<functional>

1.包装普通函数

#include <iostream>
using namespace std;
#include <functional>//1.函数
int add1(int a, int b)
{return a + b;
}//2.仿函数
struct Add2
{int operator()(int a, int b){return a + b;}
};int main()
{//1.包装函数指针function<int(int, int)> f1;f1 = add1;cout << f1(1, 2) << endl;//2.包装仿函数function<int(int, int)> f2 = Add2();//Add2():匿名对象cout << f2(1, 2) << endl;//3.包装lambda表达式function<int(int, int)> f3 = [](const int a, const int b) {return a + b; };cout << f3(1, 2) << endl;return 0;
}

在这里插入图片描述

2.包装成员函数

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

可见:包装器的本质就是对各种可调用对象进行类型的统一

3.包装器的另一个应用:统一模板实例化的类型

在这里插入图片描述

//传入该函数模板的第一个参数可以是任意可调用对象:函数指针,仿函数,lambda表达式
//func中定义了静态变量count,每次调用时打印count的值和地址,可以用来判断多次调用时所调用的是否是用一个func函数
template<class A,class T>
T func(A a, T b)
{static int count = 0;cout << "count值: " << ++count << endl;cout << "count地址: " << &count << endl;return a(b);
}//传入第二个参数相同的类型,但是传入的可调用对象的类型是不同的
//那么在编译阶段该模板函数就会被实例化多次
struct Func1
{double operator()(double d){return d / 2;}
};double func2(double d)
{return d / 2;
}int main()
{//函数cout << func(func2, 2.2) << endl;//仿函数cout << func(Func1(), 2.2) << endl;//lambda表达式cout << func([](double d)->double {return d / 2; }, 2.2) << endl;return 0;
}

在这里插入图片描述
尽管三次调用传入的可调用对象的类型不同,
但是这三次调用对象的返回值和形参类型相同

因此我们就可以使用包装器来对这三个不同的可调用对象来进行包装,此时就可以只实例化一份func函数了
在这里插入图片描述

4.小小总结

function包装器可以统一可调用对象的类型
包装后明确了可调用对象的返回值和形参类型,更加方便使用者进行使用

三.bind包装器

1.概念

在这里插入图片描述
头文件:functional

2.bind包装器绑定固定参数

bind绑定可以跟function包装器集合使用
在这里插入图片描述

1.无意义绑定

int Add(int a, int b)
{return a + b;
}int main()
{//绑定函数Add,参数分别由调用func1的第一,二参数来指定function<int(int, int)> func1 = bind(Add, placeholders::_1, placeholders::_2);cout << func1(1, 2) << endl;return 0;
}

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

2.绑定固定参数

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

3.bind包装器调整传参顺序

我们以Sub类为例
在这里插入图片描述
在这里插入图片描述
同理,普通函数也是如此
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

以上就是C++11 lambda表达式和包装器的全部内容,希望能对大家有所帮助!

http://www.lryc.cn/news/361172.html

相关文章:

  • 3. MySQL 数据表的基本操作
  • Linux命令篇(一):文件管理部分
  • IP协议1.0
  • 源码编译安装LNMP
  • 安装Chrome扩展程序来 一键禁用页面上的所有动画和过渡。有那些扩展程序推荐一下
  • 读人工智能时代与人类未来笔记19_读后总结与感想兼导读
  • 个人影响力
  • OBS实现多路并发推流
  • JDK环境配置、安装
  • 莱富康压缩机的选型软件介绍
  • Pr 2024下载安装,Adobe Premiere专业视频编辑软件安装包获取!
  • MySQL事务与MVCC
  • 【数据结构】链式二叉树详解
  • PHP面向对象编程总结
  • linux中的“->“符号
  • MySql 数据类型选择与优化
  • HTML静态网页成品作业(HTML+CSS)——家乡常德介绍网页(1个页面)
  • 【ARMv7-A】——CP15 协处理器
  • 学习笔记:(2)荔枝派Nano开机显示log(全志F1C200S)
  • Qt——升级系列(Level Two):Hello Qt 程序实现、项目文件解析、
  • VUE阻止浏览器记住密码若依CLOUD(INPUT框密码替换圆点)
  • GPT-4o:人工智能新纪元的启航者
  • CSRF跨站请求伪造漏洞
  • 【Linux】System V 消息队列(不重要)
  • label标签
  • vruntime
  • !力扣 108. 将有序数组转换为二叉搜索树
  • 13、matlab使用switch case语句实现两个数字的加减乘除运算以及数据的输入输出(可以设置计算次数)
  • 数学建模 —— 聚类分析(3)
  • java —— 匿名内部类与 Lambda 表达式