C++tuple类型
tuple 类型
tuple是类似pair的模板。
- 每个pair的成员类型都不相同,但每个pair都恰好有两个成员。
- 不同tuple类型的成员类型也不相同,但一个tuple可以有任意数量的成员。
每个确定的tuple类型的成员数目是固定的,但一个tuple类型的成员数目可以与另一个tuple类型不同。
当我们希望将一些数据组合成单一对象,但又不想麻烦地定义一个新数据结构来表示这些数据时,tuple是非常有用的。
tuple类型及其伴随类型和函数都定义在tuple头文件中。
tuple<T1, T2, ..., Tn> t; | t是一个tuple,成员数为n,第i个成员的类型为Ti.所有成员都进行值初始化 |
tuple<T1, T2, ...,Tn>t(v1,v2, ..., vn); | t是一个tuple,成员类型为T1...Tn,每个成员用对应的初始值vi进行初始化。此构造函数是explicit的 |
make_tuple(v1, v2, ..., vn) | 返回一个用给定初始值初始化的 tuple。tuple 的类型从初始值的类型推断 |
t1==t2 t1!=t2 | 当两个 tuple具有相同数量的成员且成员对应相等时,两个tuple相等。 这两个操作使用成员的==运算符来完成。一旦发现某对成员不等,接下来的成员就不用比较了 |
t1 relop t2 | tuple的关系运算使用字典序。 两个tuple必须具有相同数量的成员。使用<运算符比较t1的成员和t2中的对应成员 |
get<i>(t) | 返回t的第i个数据成员的引用;如果t是一个左值,结果是一个左值引用;否则,结果是一个右值引用。tuple的所有成员都是public的 |
tuple_size<tupleType>::value | 一个类模板,可以通过一个tuple类型来初始化。它有一个名为value的public constexpr static数据成员,类型为size_t,表示给定tuple类型中成员的数量 |
tuple_element<i, tupleType>::type | 一个类模板,可以通过一个整型常量和一个 tuple类型tupleType>::type 来初始化。它有一个名为type的public成员,表示给定tuple类型中指定成员的类型 |
我们可以将tuple看作一个“快速而随意”的数据结构。
定义和初始化tuple
当我们定义一个tuple时,需要指出每个成员的类型;
#include<iostream>
#include<tuple>
#include<vector>
#include<list>
#include<string>
using namespace std;int main()
{tuple<string,size_t, size_t> threeD; //三个成员都设置为0tuple<string, vector<double>, int, list<int>>someval("constants", { 3.14, 2, 718 }, 42, { 0,1,2,3,4,5 });
}
当我们创建一个tuple对象时,可以使用 tuple的默认构造函数,它会对每个成员进行提供一个初始值。
也可以像本例中初始化nomeval一样,为每个成员提供一个初始值
tuple的这个构造函数是explicit的因此我们必须使用直接初始化语法:
tuple<size_t, size_t, size_t> threeD = (1, 2, 3); // 错误tuple<size_t, size_t, size_t> threeD(1, 2, 3); // 正确
类似make_pair函数,标准库定义了make_tuple函数,我们还可以用它来生成tuple对象:
auto item = make_tuple("0-999-78345-X", 3, 20.00);
类似 make_pair,make_tuple 函数使用初始值的类型来推断 tuple的类型。
在本例中,item是一个tuple,类型为 tuple<const char*, int, double>。
也就是说,上面这个和下面这个等效
tuple<const char*, int, double>item("0-999-78345-X", 3, 20.00);
//注意不能tuple<const char*, int, double>item=("0-999-78345-X", 3, 20.00);
访问 tuple的成员
一个pair总是有两个成员,这样,标准库就可以为它们命名(如,first和second)。
但这种命名方式对tuple是不可能的,因为一个tuple类型的成员数目是没有限制的。
因此,tuple的成员都是未命名的。
要访问一个tuple的成员,就要使用一个名为get的标准库函数模板。为了使用get,我们必须指定一个显式模板实参,它指出我们想要访问第几个成员。
我们传递给get一个tuple对象,它返回指定成员的引用:
auto item = make_tuple("0-999-78345-X", 3, 20.00);
auto& book = get<0>(item); //返回item的第一个成员,注意在auto后面加上&
cout << book << endl;
book = "nihao";
cout << get<0>(item) << endl;auto cnt = get<1>(item); //返回item的第二个成员auto price = get<2>(item) / cnt; // 返回item的最后一个成员
get<2>(item) *= 0.8; // 打折20号
尖括号中的值必须是一个整型常量表达式。与往常一样,我们从0开始计数,意味着get<0>是第一个成员。
如果不知道一个tuple准确的类型细节信息,可以用两个辅助类模板来查询tuple成员的数量和类型:
auto item = make_tuple("0-999-78345-X", 8, 20.00);typedef decltype(item) trans;// trans是item的类型// 返回trans类型对象中成员的数量size_t sz = tuple_size<trans>::value; //返回3cout << sz << endl;// cnt的类型与item中第二个成员相同tuple_element<1, trans>::type cnt = get<1>(item); // cnt是一个intcout << cnt << endl;
为了使用tuple size或tuple element,我们需要知道一个tuple对象的类型。与往常一样,确定一个对象的类型的最简单方法就是使用decltype。在本例中,我们使用decltype来为item类型定义一个类型别名,用它来实例化两个模板。
tuple_size有一个名为value的public static数据成员,它表示给定tuple中成员的数量。
tuple_element 模板除了一个tuple类型外,还接受一个索引值。它有一个名为type的public类型成员,表示给定tuple类型中指定成员的类型。
类似get, tuple element所使用的索引也是从0开始计数的。
关系和相等运算符
tuple的关系和相等运算符的行为类似容器的对应操作。
这些运算符逐对比较左侧tuple和右侧tuple的成员。
- 只有两个 tuple 具有相同数量的成员时,我们才可以比较它们。
- 而且,为了使用tuple的相等或不等运算符,对每对成员使用==运算符必须都是合法的;
- 为了使用关系运算符,对每对成员使用<必须都是合法的。
例如:
tuple<string, string>duo("1", "2");
tuple<size_t, size_t> twoD(1, 2);
bool b = (duo == twoD); //错误:不能比较size t和string
tuple<size_t, size_t, size_t> threeD(1, 2, 3);
b = (twoD < threeD); //错误:成员数量不同
tuple<size_t, size_t> origin(0, 0);
b = (origin < twoD); // 正确:b为true
在C++中,std::tuple
类型的比较行为由它包含的元素的比较行为决定。
默认情况下,你不能比较包含不同类型元素的 std::tuple
对象,就像在你的第一个比较中那样,一个包含 std::string
类型的 tuple
和一个包含 size_t
类型的 tuple
是不能比较的,因为 std::string
和 size_t
之间没有定义比较操作。
在你的第二个比较中,尝试比较成员数量不同的 std::tuple
对象,这同样是不允许的。std::tuple
的比较要求两个 tuple
的长度(即包含的元素的数量)必须相同。
第三个比较是正确的,因为 origin
和 twoD
都是 std::tuple<size_t, size_t>
类型,它们包含相同数量和类型的元素,因此可以进行比较。在你的例子中,origin
是 (0, 0),twoD
是 (1, 2),因此 origin < twoD
会返回 true
。
总的来说,你可以比较两个 std::tuple
对象,只要它们满足以下条件:
- 两个
tuple
的长度(元素数量)必须相同。 - 对应位置的元素类型必须能够进行比较。
如果你想比较不同类型的 tuple
,你需要自定义比较逻辑,或者使用某种方式将 tuple
中的元素转换为可以比较的类型。
但请注意,这通常不是 std::tuple
的典型用法,因为 std::tuple
主要用于将不同类型的值组合成一个单一的对象,而不是用于比较操作。如果你需要频繁地比较不同类型的值,可能需要考虑使用其他数据结构或自定义类型。
由于tuple定义了<和==运算符,我们可以将tuple序列传递给算法,并且可以在无序容器中将tuple作为关键字类型。
使用 tuple返回多个值
tuple的一个常见用途是从一个函数返回多个值。
返回tupe的函数
在C++中,std::tuple 是一个模板类,它可以将多个值组合成一个单一的对象,其中每个值都可以是任意类型。当你需要从一个函数中返回多个值时,std::tuple 是一个很好的选择,因为C++不直接支持从函数中返回多个值。
下面是一个简单的例子,演示了如何定义一个返回 std::tuple 的函数:
#include <iostream>
#include <tuple>// 定义一个返回 std::tuple 的函数
std::tuple<int, std::string, double> get_values() {int a = 10;std::string b = "Hello";double c = 3.14;// 使用 std::make_tuple 创建并返回 tuplereturn std::make_tuple(a, b, c);
}int main() {// 调用函数并获取返回的 tupleauto values = get_values();// 使用 std::get 来获取 tuple 中的特定值int int_value = std::get<0>(values);std::string str_value = std::get<1>(values);double double_value = std::get<2>(values);// 输出获取到的值std::cout << "Int: " << int_value << std::endl;std::cout << "String: " << str_value << std::endl;std::cout << "Double: " << double_value << std::endl;return 0;
}
在这个例子中,get_values 函数返回一个包含 int、std::string 和 double 类型的 std::tuple。在 main 函数中,我们调用 get_values 函数,并使用 auto 关键字来自动推断返回值的类型。然后,我们使用 std::get 函数来访问 tuple 中的每个元素,并打印它们。
注意,std::get 的第一个模板参数是元素在 tuple 中的位置(从0开始)。
使用函数返回的 tuple
在C++中,函数可以返回std::tuple,它是一个固定大小的异类容器,可以存储不同类型的元素。使用函数返回tuple是一种方便的方式来返回多个值,而不需要使用结构体或类。
以下是一个关于如何在C++中使用函数返回tuple的详细解释:
1. 包含必要的头文件
首先,你需要包含tuple和utility(为了使用std::make_tuple和std::get)的头文件。
#include <tuple>
#include <utility>
2. 定义返回tuple的函数
你可以定义一个函数,其返回类型为std::tuple,并指定tuple中元素的类型。
std::tuple<int, std::string, double> get_values() {int a = 10;std::string b = "Hello";double c = 3.14;return std::make_tuple(a, b, c);
}
在这个例子中,get_values函数返回一个包含int、std::string和double类型元素的tuple。
3. 调用函数并处理返回的tuple
调用返回tuple的函数后,你可以使用std::get来访问tuple中的每个元素。注意,std::get需要元素的索引(从0开始)或类型作为参数。
int main() {// 调用函数并获取返回的tupleauto values = get_values();// 使用std::get访问tuple中的元素int int_value = std::get<0>(values);std::string str_value = std::get<1>(values);double double_value = std::get<2>(values);// 输出值std::cout << "Int: " << int_value << std::endl;std::cout << "String: " << str_value << std::endl;std::cout << "Double: " << double_value << std::endl;return 0;
}
4. 使用结构化绑定(C++17及更高版本)
从C++17开始,你可以使用结构化绑定来更简洁地处理返回的tuple。结构化绑定允许你直接将tuple的元素绑定到变量上,而无需使用std::get。
int main() {// 调用函数并使用结构化绑定处理返回的tupleauto [int_value, str_value, double_value] = get_values();// 输出值std::cout << "Int: " << int_value << std::endl;std::cout << "String: " << str_value << std::endl;std::cout << "Double: " << double_value << std::endl;return 0;
}
在这个例子中,结构化绑定自动将tuple中的元素绑定到int_value、str_value和double_value变量上,使代码更加简洁易读。
注意事项
- 确保你的编译器支持C++11或更高版本,因为std::tuple是在C++11中引入的。结构化绑定则需要C++17或更高版本的支持。
- 当处理复杂的tuple或需要在多个地方访问其元素时,考虑使用命名元组库(如Boost.Hana或std::apply与lambda表达式)以提高代码的可读性和可维护性。
相关文章:

C++tuple类型
tuple 类型 tuple是类似pair的模板。 每个pair的成员类型都不相同,但每个pair都恰好有两个成员。不同tuple类型的成员类型也不相同,但一个tuple可以有任意数量的成员。 每个确定的tuple类型的成员数目是固定的,但一个tuple类型的成员数目可…...
亚远景科技-浅谈ASPICE标准和ASPICE认证/评估
ASPICE(Automotive SPICE)是一种针对汽车行业的软件开发过程的评估模型,它旨在帮助汽车制造商和供应商提高软件开发过程的能力和质量,从而提升产品的质量、安全性和效率。 ASPICE标准涵盖了软件开发的各个阶段和活动,…...
PHP性能提升方案
一、背景与介绍 PHP语言开发效率高,特别应用于适合中小型项目,对于创业初期敏捷开发验证项目可行性或者Demo演示绝对占据优势。 但是随着现在Web应用的复杂性,针对项目要适应高并发、高流量的访问特性,PHP确实在性能方面相对Go、J…...

关系(二)利用python绘制热图
关系(二)利用python绘制热图 热图 (Heatmap)简介 热图适用于显示多个变量之间的差异,通过颜色判断彼此之间是否存在相关性。 快速绘制 基于seaborn import seaborn as sns import pandas as pd import numpy as np i…...
P8597 [蓝桥杯 2013 省 B] 翻硬币
# [蓝桥杯 2013 省 B] 翻硬币 ## 题目背景 小明正在玩一个“翻硬币”的游戏。 ## 题目描述 桌上放着排成一排的若干硬币。我们用 * 表示正面,用 o 表示反面(是小写字母,不是零),比如可能情形是 **oo***oooo&#x…...

主流公链 - Fantom
Fantom:高性能的区块链协议 Fantom是一种开创性的区块链协议,旨在革新去中心化应用和数字金融领域 技术特点 共识机制 Lachesis协议:Fantom使用了Lachesis协议作为其共识算法。Lachesis是一种 异步拜占庭容错(ABFT)共…...

vue-quill-editor 富文本编辑器(可上传视频图片),组件挂载的方式实现
1.安装 npm install vue-quill-editor --save npm install quill-image-drop-module --save npm install quill-image-resize-module --save2.在组件下面新增组件 QlEditor (1)index.vue <template><div><div idquillEditorQiniu><!-- 基于element…...
入门编程第一步,从记住这些单词开始
** 入门编程第一步,从记住这些单词开始 ** 2023-10-18 一、交互式环境与 print 输出 1、print : 打印/输出 2、coding : 编码 3、syntax : 语法 4、error : 错误 5、invalid : 无效 6、idenfifier : 名称/标识符 7、character : 字符 二、字符串的操作&#x…...

[C++]使用OpenCV去除面积较小的连通域
这是后期补充的部分,和前期的代码不太一样 效果图 源代码 //测试 void CCutImageVS2013Dlg::OnBnClickedTestButton1() {vector<vector<Point> > contours; //轮廓数组vector<Point2d> centers; //轮廓质心坐标 vector<vector<Point&…...
vscode连接不上,终端ssh正常,一直输入密码正确但是无法登录
若是之前链结果突然等不上,使用第一个链接 若是第一次链接连不上,先使用第二个链接,在使用第一个链接 原因:原因是服务器端的wget命令不能使用,vscode需要服务器端下载个文件,无法下载就导致了如上的错误…...

Hive on Spark 配置
目录 1 Hive 引擎简介2 Hive on Spark 配置2.1 在 Hive 所在节点部署 Spark2.2 在hive中创建spark配置文件2.3 向 HDFS上传Spark纯净版 jar 包2.4 修改hive-site.xml文件2.5 Hive on Spark测试2.6 报错 1 Hive 引擎简介 Hive引擎包括:MR(默认)…...
ROS 基本
ROS创建自己的功能包 ROS中工作空间(workspace)是一个存放工程开发相关文件的文件夹,其中有四个文件夹。 src:代码空间(Source Space)build:编译空间(Build Space)devel:开发空间(Development Space)install:安装空间(Install Space) OK接下来创作工作空间&#…...

Pygame基础9-射击
简介 玩家用鼠标控制飞机(白色方块)移动,按下鼠标后,玩家所在位置出现子弹,子弹匀速向右飞行。 代码 没有什么新的东西,使用两个精灵类表示玩家和子弹。 有一个细节需要注意,当子弹飞出屏幕…...

Ps:颜色查找
颜色查找 Color Lookup命令通过应用预设的 LUT 来改变图像的色彩和调性,从而为摄影师和设计师提供了一种快速实现复杂色彩调整的方法,广泛应用于颜色分级、视觉风格的统一和创意色彩效果的制作。 Ps菜单:图像/调整/颜色查找 Adjustments/Colo…...

vue3+vite 模板vue3-element-admin框架如何关闭当前页面跳转 tabs
使用模版: 有来开源组织 / vue3-element-admin 需要关闭的.vue 页面增加以下方法 //setup 里import {LocationQuery, useRoute, useRouter} from "vue-router"; const router useRouter(); function close() {console.log(|--router.currentRoute.value, router.cur…...

JavaScript 对象管家 Proxy
JavaScript 在 ES6 中,引入了一个新的对象类型 Proxy,它可以用来代理另一个对象,并可以在代理过程中拦截、覆盖和定制对象的操作。Proxy 对象封装另一个对象并充当中间人,其提供了一个捕捉器函数,可以在代理对象上拦截…...

Qt + Vs联合开发
Qt + Vs联合开发 文章目录 Qt + Vs联合开发环境说明VS+Qt安装注意事项QtCreator msvc编译器配置Visual Studio 2019 + Qt 5.12.10Visual Studio 2015 + Qt5.12.10VsQt环境配置安装插件 Qt Visual Studio Tools插件配置Qt创建项目Vs创建Qt项目VsQt工程转换Vs工程转Qt工程Qt工程转…...

开源知识库平台Raneto--使用Docker部署Raneto
文章目录 一、Raneto介绍1.1 Raneto简介1.2 知识库介绍 二、阿里云环境2.1 环境规划2.2 部署介绍 三、环境检查3.1 检查Docker服务状态3.2 检查Docker版本3.3 检查docker compose 版本 四、下载Raneto镜像五、部署Raneto知识库平台5.1 创建挂载目录5.2 编辑config.js文件5.3 编…...

鸿蒙原OS开发实例:【ArkTS类库单次I/O任务开发】
Promise和async/await提供异步并发能力,适用于单次I/O任务的场景开发,本文以使用异步进行单次文件写入为例来提供指导。 实现单次I/O任务逻辑。 import fs from ohos.file.fs; import common from ohos.app.ability.common;async function write(data:…...

C语言:二叉树的构建
目录 一、二叉树的存储 1.1 顺序存储 1.2 链式存储 二、二叉树的顺序结构及实现 2.1堆的概念及结构 2.2堆的构建 2.3堆的插入 2.4堆顶的删除 2.5堆的完整代码 三、二叉树的链式结构及实现 3.1链式二叉树的构建 3.2链式二叉树的遍历 3.2.1前序遍历 …...

简易版抽奖活动的设计技术方案
1.前言 本技术方案旨在设计一套完整且可靠的抽奖活动逻辑,确保抽奖活动能够公平、公正、公开地进行,同时满足高并发访问、数据安全存储与高效处理等需求,为用户提供流畅的抽奖体验,助力业务顺利开展。本方案将涵盖抽奖活动的整体架构设计、核心流程逻辑、关键功能实现以及…...
系统设计 --- MongoDB亿级数据查询优化策略
系统设计 --- MongoDB亿级数据查询分表策略 背景Solution --- 分表 背景 使用audit log实现Audi Trail功能 Audit Trail范围: 六个月数据量: 每秒5-7条audi log,共计7千万 – 1亿条数据需要实现全文检索按照时间倒序因为license问题,不能使用ELK只能使用…...

全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...

对WWDC 2025 Keynote 内容的预测
借助我们以往对苹果公司发展路径的深入研究经验,以及大语言模型的分析能力,我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际,我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测,聊作存档。等到明…...

《通信之道——从微积分到 5G》读书总结
第1章 绪 论 1.1 这是一本什么样的书 通信技术,说到底就是数学。 那些最基础、最本质的部分。 1.2 什么是通信 通信 发送方 接收方 承载信息的信号 解调出其中承载的信息 信息在发送方那里被加工成信号(调制) 把信息从信号中抽取出来&am…...
Python爬虫(二):爬虫完整流程
爬虫完整流程详解(7大核心步骤实战技巧) 一、爬虫完整工作流程 以下是爬虫开发的完整流程,我将结合具体技术点和实战经验展开说明: 1. 目标分析与前期准备 网站技术分析: 使用浏览器开发者工具(F12&…...
数据链路层的主要功能是什么
数据链路层(OSI模型第2层)的核心功能是在相邻网络节点(如交换机、主机)间提供可靠的数据帧传输服务,主要职责包括: 🔑 核心功能详解: 帧封装与解封装 封装: 将网络层下发…...

微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...
智能AI电话机器人系统的识别能力现状与发展水平
一、引言 随着人工智能技术的飞速发展,AI电话机器人系统已经从简单的自动应答工具演变为具备复杂交互能力的智能助手。这类系统结合了语音识别、自然语言处理、情感计算和机器学习等多项前沿技术,在客户服务、营销推广、信息查询等领域发挥着越来越重要…...

基于 TAPD 进行项目管理
起因 自己写了个小工具,仓库用的Github。之前在用markdown进行需求管理,现在随着功能的增加,感觉有点难以管理了,所以用TAPD这个工具进行需求、Bug管理。 操作流程 注册 TAPD,需要提供一个企业名新建一个项目&#…...