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

C++tuple类型

 tuple 类型

tuple是类似pair的模板。

  • 每个pair的成员类型都不相同,但每个pair都恰好有两个成员。
  • 不同tuple类型的成员类型也不相同,但一个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 t2tuple的关系运算使用字典序。
两个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的成员。

  1. 只有两个 tuple 具有相同数量的成员时,我们才可以比较它们。
  2. 而且,为了使用tuple的相等或不等运算符,对每对成员使用==运算符必须都是合法的;
  3. 为了使用关系运算符,对每对成员使用<必须都是合法的。

例如:

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 对象,只要它们满足以下条件:

  1. 两个 tuple 的长度(元素数量)必须相同。
  2. 对应位置的元素类型必须能够进行比较。

如果你想比较不同类型的 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变量上,使代码更加简洁易读。

注意事项

  1. 确保你的编译器支持C++11或更高版本,因为std::tuple是在C++11中引入的。结构化绑定则需要C++17或更高版本的支持。
  2. 当处理复杂的tuple或需要在多个地方访问其元素时,考虑使用命名元组库(如Boost.Hana或std::apply与lambda表达式)以提高代码的可读性和可维护性。

相关文章:

C++tuple类型

tuple 类型 tuple是类似pair的模板。 每个pair的成员类型都不相同&#xff0c;但每个pair都恰好有两个成员。不同tuple类型的成员类型也不相同&#xff0c;但一个tuple可以有任意数量的成员。 每个确定的tuple类型的成员数目是固定的&#xff0c;但一个tuple类型的成员数目可…...

亚远景科技-浅谈ASPICE标准和ASPICE认证/评估

ASPICE&#xff08;Automotive SPICE&#xff09;是一种针对汽车行业的软件开发过程的评估模型&#xff0c;它旨在帮助汽车制造商和供应商提高软件开发过程的能力和质量&#xff0c;从而提升产品的质量、安全性和效率。 ASPICE标准涵盖了软件开发的各个阶段和活动&#xff0c;…...

PHP性能提升方案

一、背景与介绍 PHP语言开发效率高&#xff0c;特别应用于适合中小型项目&#xff0c;对于创业初期敏捷开发验证项目可行性或者Demo演示绝对占据优势。 但是随着现在Web应用的复杂性&#xff0c;针对项目要适应高并发、高流量的访问特性&#xff0c;PHP确实在性能方面相对Go、J…...

关系(二)利用python绘制热图

关系&#xff08;二&#xff09;利用python绘制热图 热图 &#xff08;Heatmap&#xff09;简介 热图适用于显示多个变量之间的差异&#xff0c;通过颜色判断彼此之间是否存在相关性。 快速绘制 基于seaborn import seaborn as sns import pandas as pd import numpy as np i…...

P8597 [蓝桥杯 2013 省 B] 翻硬币

# [蓝桥杯 2013 省 B] 翻硬币 ## 题目背景 小明正在玩一个“翻硬币”的游戏。 ## 题目描述 桌上放着排成一排的若干硬币。我们用 * 表示正面&#xff0c;用 o 表示反面&#xff08;是小写字母&#xff0c;不是零&#xff09;&#xff0c;比如可能情形是 **oo***oooo&#x…...

主流公链 - Fantom

Fantom&#xff1a;高性能的区块链协议 Fantom是一种开创性的区块链协议&#xff0c;旨在革新去中心化应用和数字金融领域 技术特点 共识机制 Lachesis协议&#xff1a;Fantom使用了Lachesis协议作为其共识算法。Lachesis是一种 异步拜占庭容错&#xff08;ABFT&#xff09;共…...

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…...

入门编程第一步,从记住这些单词开始

** 入门编程第一步&#xff0c;从记住这些单词开始 ** 2023-10-18 一、交互式环境与 print 输出 1、print : 打印/输出 2、coding : 编码 3、syntax : 语法 4、error : 错误 5、invalid : 无效 6、idenfifier : 名称/标识符 7、character : 字符 二、字符串的操作&#x…...

[C++]使用OpenCV去除面积较小的连通域

这是后期补充的部分&#xff0c;和前期的代码不太一样 效果图 源代码 //测试 void CCutImageVS2013Dlg::OnBnClickedTestButton1() {vector<vector<Point> > contours; //轮廓数组vector<Point2d> centers; //轮廓质心坐标 vector<vector<Point&…...

vscode连接不上,终端ssh正常,一直输入密码正确但是无法登录

若是之前链结果突然等不上&#xff0c;使用第一个链接 若是第一次链接连不上&#xff0c;先使用第二个链接&#xff0c;在使用第一个链接 原因&#xff1a;原因是服务器端的wget命令不能使用&#xff0c;vscode需要服务器端下载个文件&#xff0c;无法下载就导致了如上的错误…...

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引擎包括&#xff1a;MR&#xff08;默认&#xff09…...

ROS 基本

ROS创建自己的功能包 ROS中工作空间(workspace)是一个存放工程开发相关文件的文件夹&#xff0c;其中有四个文件夹。 src:代码空间(Source Space)build:编译空间(Build Space)devel:开发空间(Development Space)install:安装空间(Install Space) OK接下来创作工作空间&#…...

Pygame基础9-射击

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

Ps:颜色查找

颜色查找 Color Lookup命令通过应用预设的 LUT 来改变图像的色彩和调性&#xff0c;从而为摄影师和设计师提供了一种快速实现复杂色彩调整的方法&#xff0c;广泛应用于颜色分级、视觉风格的统一和创意色彩效果的制作。 Ps菜单&#xff1a;图像/调整/颜色查找 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 中&#xff0c;引入了一个新的对象类型 Proxy&#xff0c;它可以用来代理另一个对象&#xff0c;并可以在代理过程中拦截、覆盖和定制对象的操作。Proxy 对象封装另一个对象并充当中间人&#xff0c;其提供了一个捕捉器函数&#xff0c;可以在代理对象上拦截…...

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提供异步并发能力&#xff0c;适用于单次I/O任务的场景开发&#xff0c;本文以使用异步进行单次文件写入为例来提供指导。 实现单次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前序遍历 …...

Python爬虫实战:研究MechanicalSoup库相关技术

一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...

多场景 OkHttpClient 管理器 - Android 网络通信解决方案

下面是一个完整的 Android 实现&#xff0c;展示如何创建和管理多个 OkHttpClient 实例&#xff0c;分别用于长连接、普通 HTTP 请求和文件下载场景。 <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas…...

Nuxt.js 中的路由配置详解

Nuxt.js 通过其内置的路由系统简化了应用的路由配置&#xff0c;使得开发者可以轻松地管理页面导航和 URL 结构。路由配置主要涉及页面组件的组织、动态路由的设置以及路由元信息的配置。 自动路由生成 Nuxt.js 会根据 pages 目录下的文件结构自动生成路由配置。每个文件都会对…...

uniapp中使用aixos 报错

问题&#xff1a; 在uniapp中使用aixos&#xff0c;运行后报如下错误&#xff1a; AxiosError: There is no suitable adapter to dispatch the request since : - adapter xhr is not supported by the environment - adapter http is not available in the build 解决方案&…...

什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南

文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/55aefaea8a9f477e86d065227851fe3d.pn…...

Maven 概述、安装、配置、仓库、私服详解

目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...

HDFS分布式存储 zookeeper

hadoop介绍 狭义上hadoop是指apache的一款开源软件 用java语言实现开源框架&#xff0c;允许使用简单的变成模型跨计算机对大型集群进行分布式处理&#xff08;1.海量的数据存储 2.海量数据的计算&#xff09;Hadoop核心组件 hdfs&#xff08;分布式文件存储系统&#xff09;&a…...

浪潮交换机配置track检测实现高速公路收费网络主备切换NQA

浪潮交换机track配置 项目背景高速网络拓扑网络情况分析通信线路收费网络路由 收费汇聚交换机相应配置收费汇聚track配置 项目背景 在实施省内一条高速公路时遇到的需求&#xff0c;本次涉及的主要是收费汇聚交换机的配置&#xff0c;浪潮网络设备在高速项目很少&#xff0c;通…...

Spring Security 认证流程——补充

一、认证流程概述 Spring Security 的认证流程基于 过滤器链&#xff08;Filter Chain&#xff09;&#xff0c;核心组件包括 UsernamePasswordAuthenticationFilter、AuthenticationManager、UserDetailsService 等。整个流程可分为以下步骤&#xff1a; 用户提交登录请求拦…...

AD学习(3)

1 PCB封装元素组成及简单的PCB封装创建 封装的组成部分&#xff1a; &#xff08;1&#xff09;PCB焊盘&#xff1a;表层的铜 &#xff0c;top层的铜 &#xff08;2&#xff09;管脚序号&#xff1a;用来关联原理图中的管脚的序号&#xff0c;原理图的序号需要和PCB封装一一…...