C++ 折叠参数包:悄然增强编程效率

news2025/1/12 21:58:39

在这里插入图片描述

前言

欢迎来到💖小K💖的💞C++专栏💞,本节将为大家带来折叠参数包的详细讲解,折叠参数包为C++模板编程提供了更加灵活和强大的工具,可以提高代码的简洁性和可读性,看完后希望对你有收获

文章目录

      • 前言
      • 一、介绍
      • 二、函数模板中使用折叠参数
        • 1、递归方式展开
        • 2、列表数据展开
        • 3、完美转发的方式展开
      • 三、类模板中使用折叠参数
        • 1、继承+模板特化的方式展开
        • 2、递归的方式展开

一、介绍

折叠参数就是一个参数包, 代表是多个未知,tuple元组就是一个折叠参数的使用

折叠参数类型:

  • typename ...Args: Args参数包的包名 ,本质是声明一个Args折叠参数类型
  • Args ...arg: 折叠参数包类型的变量
  • ...:理解为多个意思

二、函数模板中使用折叠参数

1、递归方式展开

递归方式的展开是比较好理解的,每一次调用第二个print函数就打印一次data,然后又调用自己,这时候参数包也剥离了一个参数,也就是调用自己会打印下一个data

有的同学看了下面的代码可能会疑惑,为什么会有两个函数,这是因为上面的函数为终止函数,也就是当第二个函数参数包中只有一个参数时调用第一个函数

template <typename _Ty>
void print(_Ty data) 
{
	cout << data << endl;
}
template <typename _Ty,typename ...Args>
void print(_Ty data, Args ...args) 
{
	cout << data << "\t";
	print(args...);
}

2、列表数据展开

这个的难点和重点在于initializer_list<int>{(printData(args), 0)...};,这一行代码用到了列表和逗号表达式的特性,不用说列表的每个值最后都被初始化为0,但是列表的每个值被初始化为0的时候,他们会先执行printData(args(n)),也就是会不断打印,参数包不断展开

template <typename _Ty>
void printData(_Ty data) {
	cout << data << "\t";
}
template <typename ...Args>
void printArgs(Args ...args)
{
	initializer_list<int>{(printData(args), 0)...};
	cout << endl;
}

3、完美转发的方式展开

完美转发一般是用来统一接口,也就是有许多函数,他们的参数数量、类型不同,我们把他们统一为只用函数名就可以调用该函数,且不减少其原功能

这里我们用仿函数接收一下用bind绑定的函数以及参数包,注意这里函数和参数包绑定的时候都用了完美转发

什么是完美转发呐?forword是为了解决在函数模板中,使用右值引用参数(T&&),传递右值进去以后,类型会变为左值的问题。当传入的参数是一个对象时,右值变左值就会出问题,因为左值调用拷贝构造,右值调用移动构造。本来可以用移动构造提高效率,却因为右值变成左值,调用了拷贝构造。所以我们要把它变回去!实参传的是右值,进入函数体还是右值,这就是完美转发

class Test 
{
public:
	void printk() 
	{
		if (func) func();
	}
	template <typename Func,typename ...Args>
	void connect(Func&& f, Args&& ...args)    //右值引用
	{
		func = bind(forward<Func>(f), forward<Args>(args)...);
	}
protected:
	function<void()> func;
};
void sum(int a, int b) 
{
	cout<< a + b;
}
int main() 
{
	Test test;
	test.connect(sum, 1, 2);
	test.printk();
	test.connect([](int a, int b) {cout << endl << a + b; }, 3, 8);
	test.printk();
	return 0;
}

上面的例子中通过connect绑定函数和参数包,实现统一接口的功能,通过printK函数调用

三、类模板中使用折叠参数

1、继承+模板特化的方式展开

类中实现折叠参数,前两个类是必须,对应上面的终止函数,继承的时候要写清楚public Test<Args...>,还有就是第三个类必须要一个无参构造函数,且带参数包的的构造函数初始化时要调用子类的构造函数,还有就是打印的时候要一层一层的,采用继承+模板特化就是一代一代的

template <typename ...Args>
class Test;

template<>
class Test<> {};

template <typename _Ty, typename ...Args>
class Test<_Ty, Args...> :public Test<Args...>
{
public:
	Test() {}
	Test(_Ty data, Args ...args) :data(data), Test<Args...>(args...) {}
	Test<Args...>& getObject() { return *this; }
	_Ty& getData() { return data; }
protected:
	_Ty data;
};

void testOne() 
{
	Test<string, int, double> test("fsdjf", 32, 3.23);
	cout << test.getData() << "\t" << test.getObject().getData() << "\t" << test.getObject().getObject().getData() << endl;
}

2、递归的方式展开

递归这一种就是把自己当对象调用,其它的和上面相同

template <typename ...Args>
class my_tuple;

template<>
class my_tuple<> {};

template <typename _Ty, typename ...Args>
class my_tuple<_Ty, Args...>
{
public:
	my_tuple() {}
	my_tuple(_Ty data, Args ...args) :data(data),args(args...) {}
	_Ty& getData() { return data; }
	my_tuple<Args...>& getObject() { return *this; }
protected:
	_Ty data;
	my_tuple<Args...> args;
};

void testTwo() 
{
	Test<string, int, double> test("fsdjf", 32, 3.23);
	cout << test.getData() << "\t" << test.getObject().getData() << "\t" << test.getObject().getObject().getData() << endl;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

室内外融合定位UWB信标定位方案

大家好&#xff0c;我是北京华星智控公司小智&#xff0c;今天我给大家介绍室内外融合定位系统方案&#xff0c;该方案室外采用北斗卫星定位技术室内采用UWB定位技术从而实现室内外精确定位无缝切换&#xff0c;实现室内外的融合定位。 室内外融合定位系统&#xff0c; 该方案…

Binder相关问题

Binder相关问题 1、Binder是什么&#xff1f;2、Binder有什么优势&#xff1f;3、Binder如何做到一次拷贝的&#xff1f;4、MMAP的原理是什么&#xff1f;5、Binder机制是如何跨进程的&#xff1f;6、为什么Intent不能传递大数据7、AIDL生成Java类细节8、四大组件底层的通信机制…

波奇学Linux:yum和vim

Linux三种安装方式 源代码安装 用户下载->软件源码->源码编译->可执行程序 rpm安装&#xff1a;相当于去官网下载Linux安装包 可能存在依赖项不匹配问题。 yum安装包 用户yum下载->软件安装包->可执行程序 yum是软件包管理器&#xff0c;解决安装源&…

玩客云刷Armbian带docker详细教程(附所有软件)

文章目录 介绍一.准备工作1.硬件准备2.软件准备 二.开始折腾1.烧录系统固件2.刷入系统1.准备镜像2.刷入镜像3.刷入镜像到系统 三.功能介绍1.网页终端2.设备状态3.AriaNg4.qBittorrent5.微力同步6.filebrowser7.Portainer 四.拓展1.添加Alist1.介绍2.安装3.配置3.1查看密码3.2修…

chatgpt赋能Python-pythoncidere

PythonCider&#xff1a;提供最优质的 Python 编程知识 Python 是目前最受欢迎的编程语言之一&#xff0c;特别在数据科学和人工智能方面应用广泛。如果你是一名 Python 开发者或者正在学习 Python 编程&#xff0c;PythonCider 是一个你绝对不能错过的网站。 什么是 PythonC…

JavaScript实战训练小项目 WebAPI

JavaScript实战训练小项目 文章目录 JavaScript实战训练小项目 & WebAPI1. JS操作DOM树1.1 获得HTML控件/元素标签1.2 操纵控件1.2.1 获取属性值1.2.1 修改属性值 1.3 实现一个猜数字的功能 2. JQuery3. 简单计算器4. 聚合搜索5. 表白墙 JavaScript实战训练小项目 & We…

OpenLayers入门教程汇总目录

前言 本篇作为OpenLayers入门教程的目录&#xff0c;用于整理汇总专栏所有文章&#xff0c;方便查找。 入门 Gis开发入门&#xff0c;OpenLayers、Leaflet、Maplibre-gl和Cesiumjs地图引擎介绍以及几种地图服务vms、vmts介绍vue项目集成并使用OpenLayers地图的两种方式 加…

记ABAC的落地实践

为什么使用ABAC 一般提到授权&#xff0c;我们就会想到角色&#xff08;role&#xff09;。什么样的用户拥有什么样的角色可以怎么操作什么样的资源&#xff0c;这是我们普遍使用的权限系统的模型。这里的角色实质上是包含了一组用户操作资源的规则集合。一旦角色被创建&#…

【RISC-V】执行环境

裸机程序 操作系统 虚拟化 RISC-V处理器3种模式 机器模式(M模式) 安全执行环境&#xff0c;通常运行SBI固件&#xff0c;为操作系统提供服务。 特权模式(S模式) 运行操作系统内核&#xff0c;为应用程序提供服务。 用户模式(U模式) 运行应用程序。 虚拟化新增特权模式 …

Boundary IoU:Improving Object-Centric Image Segmentation Evaluation总结笔记

Boundary IoU:Improving Object-Centric Image Segmentation Evaluation&#xff08;边界Iou&#xff1a;改进以对象为中心的图像分割评价&#xff09; 目录 一、论文出发点 二、论文核心思想 三、相关工作 四、敏感度分析 五、Boundary IoU定义和实验证明 六、应用 七…

【系统移植】uboot 通过 NFS 加载根文件系统(一) —— 网络环境配置

前面试过了SD卡烧录根文件系统&#xff0c;然后借助环境变量 bootargs 来加载SD卡中的根文件系统&#xff0c;但是实际开发需要经常对文件做增删改等操作&#xff0c;所以将根文件系统放在SD上不大合适。 因此&#xff0c;最常用的做法是把根文件系统放在NFS服务端&#xff08…

Spring事务及事务传播机制

一.事务的含义:多个操作封装在一起&#xff0c;要么同时执行成功&#xff0c;一旦有一个操作执行失败&#xff0c;那么全部执行失败。这里给大家举个例子:比如A给B转账50元&#xff0c;而B没有收到这50元&#xff0c;此时A转账B这个操作也需要进行回滚&#xff0c;恢复到A给B没…

HTTP 协议的基本格式

HTTP 什么是HTTP?HTTP是应用层的一个重要协议.它定义了浏览器怎样向服务器请求文件,以及服务器怎样把文件传送给浏览器. 我们打开浏览器,手动输入一个网址:baidu.com.那么此时浏览器就会给百度的服务器发送请求.百度服务器在返回一个html的响应. 那么我们如何学习HTTP协议呢…

【系统移植】uboot 通过 NFS 加载根文件系统(二) —— 修改环境变量 bootcmd、bootargs

上一部分已经配置好了网络环境&#xff0c;我们要修改环境变量以保证开发板可以正常启动&#xff0c;uboot 环境变量中比较重要的环境变量就是 bootcmd 和 bootargs 目录 1、修改自启动命令 — bootcmd 2、修改自启动参数 —— bootargs (1) root/dev/nfs rw (2) nfsroot …

Godot引擎 4.0 文档 - 循序渐进教程 - 创建你的第一个脚本

本文为Google Translate英译中结果&#xff0c;DrGraph在此基础上加了一些校正。英文原版页面&#xff1a; Creating your first script — Godot Engine (stable) documentation in English 创建你的第一个脚本 在本课中&#xff0c;您将编写您的第一个脚本&#xff0c;使用…

Datacom-HCIE 题库 02(10月26日更新)--含解析

单选题 1.[试题编号&#xff1a;189785] &#xff08;单选题&#xff09;如图所示&#xff0c;VTEP1上在BD20域内开启了ARP广播抑制功能&#xff0c;并且VTEP1通过 BGP EVPN路由学习到了PC2的ARP信息&#xff0c;则PC1发送的针对PC2的ARP请求&#xff0c;VIEP1在转发给VIEP2时…

设备采购信息管理系统

系列文章 任务14 设备采购信息管理系统 文章目录 系列文章一、实践目的与要求1、目的2、要求 二、课题任务三、总体设计1.存储结构及数据类型定义2.程序结构3.所实现的功能函数4、程序流程图 四、小组成员及分工五、 测试界面展示添加采购信息按编号查找采购信息按设备编号查找…

vue记录鼠标拖拽划过位置并将划过位置变色

首先 我们要做一个这样的基本组件 <template><div><!--循环遍历 List数组用当前下面当做key值然后定义了 onDragStart 鼠标拖动时触发定义 onDragEnd 拖动结束后触发定义 onDragOver 记录所有鼠标拖动经过的位置--><divclass"skeleton"v-f…

DAY 63 mysql的高级语句:存储过程

什么是存储过程 存储过程是一组为了完成特定功能的SQL语句集合。 存储过程在使用过程中是将常用或者复杂的工作预先使用SQL语句写好并用一个指定的名称存储起来&#xff0c;这个过程经编译和优化后存储在数据库服务器中。当需要使用该存储过程时&#xff0c;只需要调用它即可…

数据结构之堆详解

目录 1.什么是堆 堆的定义 结构体定义与函数接口 堆的初始化 堆的销毁 入堆 向上调整算法 大堆 出堆 向下调整算法 返回堆顶元素 判空 堆的应用 1.什么是堆 知道以上的存储方法&#xff0c;对于完全二叉树&#xff0c;有一个叫做堆的结构&#xff0c;堆本质就是一…