C++11常用知识分享(二)【可变参数模板 || lambda表达式 || 包装器】

目录

一,可变参数模板

1. 递归方法展开参数包

2. 逗号表达式展开参数包

3,可变参数模板优势

二,lambda表达式

1. lambda表达式语法

2. 注意点

三,包装器

1. bind(了解)


嗨!收到一张超美的风景图,希望你每天都能顺心 

一,可变参数模板

由于可变模版参数比较抽象,使用起来需要一定的技巧,所以这块还是比较晦涩的。现阶段呢,我们掌握一些基础的可变参数模板特性。

// Args是一个模板参数包,args是一个函数形参参数包

// 声明一个参数包Args...args,这个参数包中可以包含0到任意个模板参数。

template <class ...Args>

void ShowList(Args... args)

{}

上面的参数args前面有省略号,所以它就是一个可变模版参数,我们把带省略号的参数称为“参数包”,它里面包含了0到N(N>=0)个模版参数。我们无法直接获取参数包args中的每个参数的,只能通过展开参数包的方式来获取参数包中的每个参数,这是使用可变模版参数的一个主要特点,也是最大的难点,即如何展开可变模版参数。

1. 递归方法展开参数包

template <class T>
void func(const T& t)
{
	cout << t << endl;
	cout << endl;
}

void func()
{
	cout << 0 << endl;
}

template<class T , class ...Ags>
//void func(const T& t, Ags... args)  // 一个一个地解包参数
 void func(const T& t, Ags&&... args)    // 
{
	cout << t << " 剩余包数:" << sizeof...(args) << endl ;
	//func(args...); // 剩余参数包
	func(forward<Ags>(args)...); //...的位置有讲究,
}

int main()
{
	func();
	func(1, 'A');
	func(1, 'A', "hello word");
}

2. 逗号表达式展开参数包

template <class T>
void PrintArg(T t)  // 参数处理函数
{
 cout << t << " ";
}
//展开函数
template <class ...Args>
void ShowList(Args... args)
{
 int arr[] = { (PrintArg(args), 0)... };
 cout << endl;
}
int main()
{
 ShowList(1);
 ShowList(1, 'A');
 ShowList(1, 'A', std::string("sort"));
 return 0;
}

expand函数中的逗号表达式:(printarg(args), 0),也是按照这个执行顺序,先执行printarg(args),再得到逗号表达式的结果0。同时还用到了C++11的另外一个特性——初始化列表,通过初始化列表来初始化一个变长数组, {(printarg(args), 0)...}将会展开成((printarg(arg1),0), (printarg(arg2),0), (printarg(arg3),0), etc... ),最终会创建一个元素值都为0的数组int arr[sizeof... (Args)]。由于是逗号表达式,在创建数组的过程中会先执行逗号表达式前面的部分printarg(args)打印出参数,也就是说在构造int数组的过程中就将参数包展开了,这个数组的目的纯粹是为了在数组构造的过程展开参数包。

3,可变参数模板优势

    vector<pair<int, string>> it;
	it.emplace_back(1, "sort", 2, "sort", 3, "sort");
	it.emplace_back(1,"sort");

	it.push_back(make_pair(1, "sort"));
	it.push_back({1,"sort"});
	// 优势:emplace_back直接构造;push_back需要2次构造

 虽然只是减少了一次构造,但push_back的功能emplace_back都有,而且后者还具备一些优势。

二,lambda表达式

1. lambda表达式语法

lambda 表达式书写格式:
[capture-list] (parameters) mutable -> return-type { statement }
[capture-list] : 捕捉列表 ,该列表总是出现在 lambda 函数的开始位置, 编译器根据 [] 判断接下来的代码是否为 lambda 函数 捕捉列表能够捕捉上下文中的变量供 lambda 函数使用
(parameters) 参数列表。与 普通函数的参数列表一致 ,如果不需要参数传递,则可以连同() 一起省略
mutable :       默认情况下, lambda 函数总是一个 const 函数 mutable 可以取消其常量性。使用该修饰符时,参数列表不可省略( 即使参数为空 )
->returntype :返回值类型 。用 追踪返回类型形式声明函数的返回值类型 ,没有返回值时此部分可省略。返回值类型明确情况下,也可省略,由编译器对返回类型进行推
{statement} :函数体 。在该函数体内,除了可以使用其参数外,还可以使用所有捕获到的变量。
注意:
lambda 函数定义中, 参数列表和返回值类型都是可选部分,而捕捉列表和函数体可以为 。因此 C++11 最简单的 lambda 函数为: []{} ; lambda 函数不能做任何事情。
int main()
{
 vector<Goods> v = { { "苹果", 2.1, 5 }, { "香蕉", 3, 4 }, { "橙子", 2.2, 3 }, { "菠萝",1.5, 4 } };
 sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2){
 return g1._price < g2._price; });

 sort(v.begin(), v.end(), [](const Goods& g1, const Goods& g2){
 return g1._price > g2._price; });
 // lambda的一个优势:面对简单的仿函数,就地建立,省去了寻找仿函数那一步。


 // 传值捕捉-——是捕捉对象的拷贝
 int x = 0, y = 1;
 auto swap = [x,y]() mutable  //捕捉对象是不被修改的常性 + mutacble可以修改捕捉对象,但无法修改外部对象
 {  auto temp = x;
    x = y;
    y = temp;
 }

 // 引用捕捉——不加mutable  可以修改外部对象
 auto swap2 = [&x, &y](){
    auto temp = x;
    x = y;
    y = temp;
 }

 // 全传值捕捉 & 全引用 
   auto swaps = [=]()
// auto swaps = [&]()
{...}

}

2. 注意点

a. 父作用域指包含 lambda 函数的语句块。
b. 语法上捕捉列表可由多个捕捉项组成,并以逗号分割 。 比如:[=, &a, &b] :以引用传递的方式捕捉变量 a b ,值传递方式捕捉其他所有变量[&, a, this] :值传递方式捕捉变量 a this ,引用方式捕捉其他变量。
c. 捕捉列表不允许变量重复传递,否则就会导致编译错误 。比如:[=, a] = 已经以值传递方式捕捉了所有变量,捕捉 a 重复。
d. 在块作用域以外的 lambda 函数捕捉列表必须为空
e. 在块作用域中的 lambda 函数仅能捕捉父作用域中局部变量,捕捉任何非此作用域或者非局部变量都会导致编译报错。
f. lambda 表达式之间不能相互赋值 ,即使看起来类型相同。

总之,lamda本质上是一个仿函数

三,包装器

function包装器 也叫作适配器。C++中的function本质是一个类模板,也是一个包装器。
std::function在头文件<functional>
// 类模板原型如下
template <class T> function;     // undefined

template <class Ret, class... Args>
class function<Ret(Args...)>;

模板参数说明:
Ret: 被调用函数的返回类型
Args…:被调用函数的形参
包装器本质上是创建仿函数,进行调用函数

例子:

    int func(int x, int y)
    {
		return	x + y;
	}

	class fun
	{
	public:
		int operator()(int x, int y)
		{
		return x * y;
		}
	};

	class plusi
	{
	public:
		int Add_plus(int a, int b)
		{
			return a + b;
		}

	    static int Add(int a, int b)
		{
			return a + b;
		}
	};

	int main()
	{
		function<int(int, int)> f1 = func;  // 函数指针
		function<int(int, int)> f2 = fun();  // 仿函数
		function<int(int, int)> f3 = [](int x, int y) {return x + y; };  // lambda表达式
		function<int(int, int)> f4 = plusi::Add;   // 静态成员函数
		function<int(plusi,int, int)> f5 = &plusi::Add_plus; 	//Add_plus参数中存在隐藏的this指针,但添加的不是plusi指针。
		// 为什么是 plusi,而不是plusi*, 原因:f5(plusi(),2, 3), 如果是指针,那么无法使用匿名对象。


		// 且f1,f2,f3类型相同

        // 利用键值对进行映射
		map < string, function<int(int, int)>>mp;
		mp["函数指针"] = f1;
		mp["仿函数"] = f2;
		mp["lambda表达式"] = f3;

		cout << f1(1, 2) << endl;
		cout << f2(1, 2) << endl;
		cout << f3(1, 2) << endl;
		cout << f5(plusi(),2, 3) << endl;

		return 0;
	}
	}

尝试使用包装器来解题: 

150. 逆波兰表达式求值 - 力扣(LeetCode)

1. bind(了解)

std::bind函数定义在头文件中,是一个函数模板,它就像一个函数包装器(适配器),接受一个可调用对象(callable object),生成一个新的可调用对象来“适应”原对象的参数列表。一般而言,我们用它可以把一个原本接收N个参数的函数fn,通过绑定一些参数,返回一个接收M个(M可以大于N,但这么做没什么意义)参数的新函数。同时,使用std::bind函数还可以实现参数顺序调整等操作。

功能: 1. 调整参数顺序(了解) 2.调整参数个数(使用较多)

int func(int x, int y)
    {
		return	x / y;
	}

   class print
   {
   public:
	   int multiply(int x, int y)
	   {
		   return x * y;
	   }
	};

   int main()
   {
	   // 1.调整形参顺序 
	   auto Pfunc1 = bind(func, placeholders::_1, 2); // 交换形参,_1存在于placeholders命名空间中
	   cout << Pfunc1(10, 2) << endl; // 5  其中第三个参数为2,可以理解为Pfunc1函数,第三参数一直是2,我叫他“绑死”“显示传递”
	   auto Pfunc2 = bind<int>(func, placeholders::_2, placeholders::_1); // 重新修改形参的新函数
	   cout << Pfunc2(2, 10) << endl; // 0.5

	   // 2.调整形参个数
	   map<string, function<int(int, int)>> mp;
	   mp["调整形参顺序"] = Pfunc1;

	   function<int(print, int,int)> Pfunc3 = &print::multiply; // 三个参数无法载入map中
	   auto Pfunc4 = bind(Pfunc3, print(), placeholders::_1, placeholders::_2); //通过bind显示传递(绑死),减少传参个数
	   mp["调整形参个数"] = Pfunc4;

	   return 0;
   }

结语

   本小节就到这里了,感谢小伙伴的浏览,如果有什么建议,欢迎在评论区评论,如果给小伙伴带来一些收获请留下你的小赞,你的点赞和关注将会成为博主创作的动力

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/428592.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C向C++的一个过渡

思维导图 输入输出&#xff0c;以及基础头文件 在c语言中我们常用scanf("%d",&n);和printf("%d\n",n);来输出一些变量和常量&#xff0c;在C中我们可以用cin;和cout;来表示输入输出。 在C语言中输入输出有头文件&#xff0c;在C也有头文件&#xff0…

#WEB前端(CCS选择器)

1.实验&#xff1a;CCS选择器 2.IDE&#xff1a;VSCODE 3.记录&#xff1a; 子代选择器、后代选择器、相邻兄弟选择器、类选择器、伪元素选择器&#xff08;鼠标悬停&#xff09;、ID选择器、调用选择器&#xff08;全选&#xff09; 4.代码&#xff1a; <!DOCTYPE html…

Vue.js 实用技巧:深入理解 Vue.set 方法

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

JMeter常用函数整理

"_csvRead"函数 csvRead函数是从外部读取参数&#xff0c;csvRead函数可以从一个文件中读取多个参数。 下面具体讲一下如何使用csvread函数&#xff1a; 1.新建一个csv或者text文件&#xff0c;里面保存要读取的参数&#xff0c;每个参数间用逗号相隔。每行表示每一组…

MATLAB:Image Processing Toolbox工具箱入门实战

目录 1.基本图像导入、处理和导出 2.实战项目一&#xff1a;利用imfindcircles()函数检测和测量图像中的圆形目标 1.基本图像导入、处理和导出 Basic Image Import, Processing, and Export- MATLAB & SimulinkThis example shows how to read an image into the worksp…

BUUCTF---[极客大挑战 2019]Http1

1.题目描述&#xff0c;在地址框输入下面的网址 2.来到页面&#xff0c;ctrlu查看源码&#xff0c;仔细观察会看到一个.php的跳转页面 3.点进去页面提示It doesnt come from https://Sycsecret.buuoj.cn 4.页面提示它不是来源于这个网址&#xff0c;我们需要用bp抓包对数据进行…

从0到1全流程使用 segment-anything

从0到1全流程使用 segment-anything 一、安装 anaconda 一、下载 anaconda 二、以管理员身份运行安装 1、勾选 Just Me 2、统一安装路径(后续 python 等包也安装至此目录) 3、勾选 add to path 然后安装即可。 三、修改 Anaconda 默认路径及默认缓存路径 Anaconda 默认下…

神经网络3-时间卷积神经网络

在深度学习的知识宝库中&#xff0c;卷积神经网络&#xff08;CNN&#xff09;广泛应用于视觉&#xff0c;视频等二维或者多维的图像领域。卷积网络具有深度&#xff0c;可并行等多种优良特性&#xff0c;那么这种技术是否可以应用于解单维度的时间序列问题呢&#xff1f;本文介…

基于Springboot的助农管理系统(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的助农管理系统&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&…

UTONMOS元宇宙游戏发展趋势是什么?

UTONMOS元宇宙游戏的发展趋势包括以下几个方面&#xff1a; 更加真实的体验&#xff1a;随着技术的进步&#xff0c;UTONMOS元宇宙游戏将提供更加逼真的视觉、听觉和触觉体验&#xff0c;让玩家更加身临其境。 社交互动&#xff1a;UTONMOS元宇宙游戏将越来越注重社交互动&am…

Linux系统宝塔面板搭建Typecho博客并实现公网访问本地网站【内网穿透】

文章目录 前言1. 安装环境2. 下载Typecho3. 创建站点4. 访问Typecho5. 安装cpolar6. 远程访问Typecho7. 固定远程访问地址8. 配置typecho 前言 Typecho是由type和echo两个词合成的&#xff0c;来自于开发团队的头脑风暴。Typecho基于PHP5开发&#xff0c;支持多种数据库&#…

Windows服务器:通过nginx反向代理配置HTTPS、安装SSL证书

先看下效果&#xff1a; 原来的是 http&#xff0c;配置好后 https 也能用了&#xff0c;并且显示为安全链接。 首先需要 SSL证书 。 SSL 证书是跟域名绑定的&#xff0c;还有有效期。 windows 下双击可以查看相关信息。 下载的证书是分 Apache、IIS、Tomcat 和 Nginx 的。 我…

9.10目标和(LC494-M)

算法&#xff1a; 加法的绝对值的集合left 减法的绝对值的集合right nums集合的总和sum 这里的left和right都是绝对值&#xff1a; leftrightsum → rightsum-left left-righttarget → left-(sum-left) target → left (target sum)/2 &#xff0c;target …

最新AI系统ChatGPT网站H5系统源码,支持Midjourney绘画

一、前言 SparkAi创作系统是基于ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;那么如何搭建部署AI创作ChatGPT&#xff1f;小编这里写一个详细图文教程吧。已支持GPT…

python封装,继承,复写详解

目录 1.封装 2.继承 复写和使用父类成员 1.封装 class phone:__voltage 0.5def __keepsinglecore(self):print("单核运行")def callby5g(self):if self.__voltage > 1:print("5g通话开启")else:self.__keepsinglecore()print("不能开启5g通…

xshell安装java/jdk

1.下载jdk wget https://download.java.net/java/GA/jdk11/13/GPL/openjdk-11.0.1_linux-x64_bin.tar.gz 2.解压jdk安装包 tar -zxvf openjdk-11.0.1_linux-x64_bin.tar.gz 其中第三步 编辑 ~/.bashrc 或 ~/.bash_profile 文件 打开vim文本编辑器 vim ~/.bash_profile export …

TT-100K数据集

TT-100K数据集 TT100K数据集是由清华大学和腾讯联合实验室整理并公布的一个大型交通标志数据集。已整理好由xml格式和txt格式。共6105张图片。 有偿分享。可以加我qq&#xff1a;2638351996。注明来意&#xff01;&#xff01;&#xff01;&#xff01;

深入理解Python递归:注意事项、示例及应用场景

文章目录 一、递归的注意事项二、Python代码示例三、使用场景及代码运行结果四、递归的其他应用场景其他示例 五、总结 递归是编程中的一种强大的技术&#xff0c;它允许函数调用自身来解决问题。在Python中&#xff0c;递归被广泛应用&#xff0c;尤其是在处理数据结构&#x…

算法沉淀——动态规划之01背包问题(leetcode真题剖析)

算法沉淀——动态规划之01背包问题 01.【模板】01背包02.分割等和子集03.目标和04.最后一块石头的重量 II 01背包问题是一类经典的动态规划问题&#xff0c;通常描述为&#xff1a;有一个固定容量的背包&#xff0c;以及一组物品&#xff0c;每件物品都有重量和价值&#xff0c…

大数据核心技术概论

大数据核心技术概述 大数据基石三大论文&#xff1a;GFS&#xff08;Hadoop HDFS&#xff09;、BigTable&#xff08;Apache HBase&#xff09;、MapReduce&#xff08;Hadoop MapReduce&#xff09;。 搜索引擎的核心任务&#xff1a;一是数据采集&#xff0c;也就是网页的爬…