C++中神奇的tuple:详解使用技巧和实例解析

C++中神奇的tuple:详解使用技巧和实例解析

  • 一、tuple的基本概念
  • 二、tuple基础知识
    • 2.1、tuple的创建和初始化
    • 2.2、tuple的成员访问
    • 2.3、效果展示
    • 2.4、tupe的成员函数、非成员函数 及 辅助类
  • 三、tuple高级应用技巧
    • 3.1、tuple的结构化绑定
    • 3.2、tuple的运算符重载
    • 3.3、tuple的嵌套和组合
  • 四、tuple实例解析
  • 五、tuple的性能和适用场景分析
  • 总结

一、tuple的基本概念

在C++中,tuple是一种数据结构,用于将多个值组合在一起,形成一个有序的元组。每个值在tuple中都有一个对应的索引,可以通过索引来访问和操作其中的值。

作用:

  1. tuple将多个值组合在一起,形成一个整体。这些值可以是不同的数据类型,例如整数、浮点数、字符串等。

  2. tuple中的值是有序排列的,每个值都有一个对应的位置索引,可以通过索引来访问和操作其中的值。

  3. 可以将tuple中的值解构出来,分别赋值给不同的变量,方便对其中的值进行单独处理。

  4. 在函数中可以使用tuple作为返回类型,从而方便返回多个值,而不需要使用指针或引用参数。

  5. C++17引入了结构化绑定的特性,可以方便的从tuple中提取出其中的值并进行使用,在代码书写上更加简洁和可读。

在这里插入图片描述

二、tuple基础知识

C++中的tuple是一个标准库类模板,用于存储固定数量的异类对象。允许将多个对象捆绑成一个单一的对象,并且可以轻松地从中提取值或者通过结构化绑定将其解构到不同的变量中。tuple提供了一个通用的数据结构,可以保存不同类型的元素,并通过下标或者std::get函数来访问其中的值。

C++11引入了最初的tuple实现, C++17进一步扩展了其功能,增加了结构化绑定的支持,大大提高了tuple在实际应用中的便利性。

2.1、tuple的创建和初始化

(1)直接初始化:

#include <tuple>
#include <string>int main() {// 直接初始化一个tuplestd::tuple<int, double, std::string> myTuple(10, 3.14, "Hello");return 0;
}

(2)使用std::make_tuple函数进行初始化:

#include <tuple>
#include <string>int main() {// 使用make_tuple函数创建一个tupleauto myTuple = std::make_tuple(10, 3.14, "Hello");return 0;
}

(3)使用std::tie进行结构化绑定的初始化:

#include <tuple>
#include <string>int main() {int a;double b;std::string c;std::tie(a, b, c) = std::make_tuple(10, 3.14, "Hello");return 0;
}

(4)使用std::forward_as_tuple进行创建和初始化:

#include <tuple>
#include <string>int main() {auto myTuple = std::forward_as_tuple(10, 3.14, "Hello");return 0;
}

2.2、tuple的成员访问

(1)使用 std::get 函数按索引访问元素:

#include <iostream>
#include <tuple>int main() {std::tuple<int, double, std::string> myTuple(10, 3.14, "Hello");int intValue = std::get<0>(myTuple);double doubleValue = std::get<1>(myTuple);std::string stringValue = std::get<2>(myTuple);std::cout << "Int value: " << intValue << std::endl;std::cout << "Double value: " << doubleValue << std::endl;std::cout << "String value: " << stringValue << std::endl;return 0;
}

(2)使用 std::tie 进行结构化绑定,将 tuple 成员绑定到指定的变量:

#include <iostream>
#include <tuple>int main() {int a;double b;std::string c;std::tuple<int, double, std::string> myTuple(10, 3.14, "Hello");std::tie(a, b, c) = myTuple;std::cout << "Int value: " << a << std::endl;std::cout << "Double value: " << b << std::endl;std::cout << "String value: " << c << std::endl;return 0;
}

(3)使用结构化绑定(C++17):

#include <iostream>
#include <tuple>int main() {std::tuple<int, double, std::string> myTuple(10, 3.14, "Hello");auto [intValue, doubleValue, stringValue] = myTuple;std::cout << "Int value: " << intValue << std::endl;std::cout << "Double value: " << doubleValue << std::endl;std::cout << "String value: " << stringValue << std::endl;return 0;
}

注意,std::forward_as_tuple的初始化方式是无法使用结构化绑定访问元素的。

2.3、效果展示

示例:

#include <tuple>
#include <iostream>
# include <string>
int main(int n,char ** args)
{// 直接初始化:std::tuple<int, double, std::string> mtuple(1,2.0,"3a");// 使用`std::make_tuple`函数进行初始化:auto mtuple2 = std::make_tuple(11,22.0,"3aa");// 使用`std::tie`进行结构化绑定的初始化:int a;double b;std::string s;std::tie(a,b,s)=std::make_tuple(111,222.0,"3aaa");// 使用`std::forward_as_tuple`进行创建和初始化:auto mtuple3=std::forward_as_tuple(1111,2222.0,"3aaaa");// 使用 `std::get` 函数按索引访问元素:std::cout<<"使用 `std::get` 函数按索引访问元素:"<<std::endl;int oa=std::get<0>(mtuple);double ob=std::get<1>(mtuple);std::string os=std::get<2>(mtuple);std::cout<<"oa="<<oa<<",ob="<<ob<<",os="<<os<<std::endl;std::cout<<"使用 std::tie 进行结构化绑定,将 tuple 成员绑定到指定的变量:"<<std::endl;std::cout<<"a="<<a<<",b="<<b<<",s="<<s<<std::endl;std::cout<<"使用结构化绑定(C++17):"<<std::endl;// 注意,std::forward_as_tuple的初始化方式是无法使用结构化绑定访问元素的。auto [intVal,doubleVal,strVal] = mtuple2;std::cout<<"intVal="<<intVal<<",doubleVal="<<doubleVal<<",strVal="<<strVal<<std::endl;return 0;
}

输出:

使用 `std::get` 函数按索引访问元素:
oa=1,ob=2,os=3a
使用 std::tie 进行结构化绑定,将 tuple 成员绑定到指定的变量:
a=111,b=222,s=3aaa
使用结构化绑定(C++17):
intVal=11,doubleVal=22,strVal=3aa

2.4、tupe的成员函数、非成员函数 及 辅助类

成员函数描述
constructor(C++11)构造一个新的“元组”(公共成员函数)
operator=(C++11)将一个“元组”的内容分配给另一个(公共成员函数)
swap (C++11)交换两个元组的内容(公共成员函数)
非成员函数描述
std::tuple_size返回 tuple 中元素的数量
std::tuple_element返回 tuple 中指定索引的元素类型
std::get通过索引访问 tuple 的元素
std::tie将 tuple 的元素绑定到指定的变量
std::make_tuple创建 tuple
std::forward_as_tuple创建 tuple,保留变量的类型(包括引用)
std::tuple_cat连接两个或多个 tuple
std::swap交换两个 tuple 的内容
辅助类描述
std::pair包含两个值的不同类型的固定大小集合
std::tuple_sizetuple 类型的长度
std::tuple_elementtuple 类型的元素类型

这些成员函数、非成员函数和辅助类提供了丰富的功能,可以用于创建、访问和操作 tuple,使得 tuple 在 C++ 中成为一个强大且灵活的数据结构。

三、tuple高级应用技巧

3.1、tuple的结构化绑定

因为tuple的结构化绑定是一种非常方便的技巧,所以这里再次强调一下。在 C++17 中引入了结构化绑定(structured bindings),可以用来将 tuple 或其他数据结构中的元素绑定到多个变量中,而无需显式地通过索引进行访问。这个技巧可以让代码更加清晰和易读,特别是在处理多个返回值或者复杂的数据结构时非常有用。

(1)对于 tuple 结构化绑定的使用(再次强调):

std::tuple<int, double, std::string> myTuple(10, 3.14, "Hello");auto [intValue, doubleValue, stringValue] = myTuple;std::cout << "Int value: " << intValue << std::endl;
std::cout << "Double value: " << doubleValue << std::endl;
std::cout << "String value: " << stringValue << std::endl;

(2)结构化绑定也可以用于返回多个值的函数:

std::tuple<int, double> getData() {return std::make_tuple(10, 3.14);
}auto [value1, value2] = getData();std::cout << "Value 1: " << value1 << std::endl;
std::cout << "Value 2: " << value2 << std::endl;

(3)结构化绑定还可以用于 STL 容器中的元素访问:

std::map<int, std::string> myMap = {{1, "One"}, {2, "Two"}, {3, "Three"}};for (const auto& [key, value] : myMap) {std::cout << "Key: " << key << ", Value: " << value << std::endl;
}

(4)使用结构化绑定处理复杂的数据结构:

std::unordered_map<std::string, std::pair<int, double>> data = {{"item1", {10, 3.14}}, {"item2", {20, 6.28}}};for (const auto& [key, value] : data) {int intValue = value.first;double doubleValue = value.second;std::cout << "Key: " << key << ", Int value: " << intValue << ", Double value: " << doubleValue << std::endl;
}

结构化绑定的引入让 tuple对其他数据结构的处理变得更加简洁、清晰和灵活。

3.2、tuple的运算符重载

(1)比较运算符重载(如==、!=、<、>等),实现自定义的比较行为。例如对 tuple 元素的比较:

bool operator==(const std::tuple<int, double>& t1, const std::tuple<int, double>& t2) {return std::get<0>(t1) == std::get<0>(t2) && std::get<1>(t1) == std::get<1>(t2);
}

(2)算术运算符重载如+、-、*、/等),实现对 tuple 的算术操作。例如对两个 tuple 进行加法运算:

std::tuple<int, double> operator+(const std::tuple<int, double>& t1, const std::tuple<int, double>& t2) {return std::make_tuple(std::get<0>(t1) + std::get<0>(t2), std::get<1>(t1) + std::get<1>(t2));
}

(3)流操作符重载(<<>>)。实现对 tuple 的输入输出操作,使得可以直接使用流操作符对 tuple 进行输入输出:

template <typename... Args>
std::ostream& operator<<(std::ostream& os, const std::tuple<Args...>& t) {os << "(";std::apply([&os](const auto&... args) {((os << args << ", "), ...);}, t);os << ")";return os;
}

3.3、tuple的嵌套和组合

在 C++ tuple 嵌套和组合是为处理复杂的数据结构提供了灵活性和便利性。

(1)嵌套 tuple:将多个 tuple 组合成一个更大的 tuple,以便对多个数据结构进行整体操作,主要使用tuple_cat函数。
例如,将两个 tuple 进行组合:

std::tuple<int, double> tuple1 = std::make_tuple(10, 3.14);
std::tuple<std::string, char> tuple2 = std::make_tuple("hello", 'A');auto combinedTuple = std::tuple_cat(tuple1, tuple2);
// combinedTuple 的类型为 std::tuple<int, double, std::string, char>

(2)元组中嵌套元组:在 tuple 中嵌套另一个 tuple,以实现更复杂的数据结构。例如,创建一个包含两个 tuple 的元组:

std::tuple<int, std::tuple<double, std::string>> nestedTuple = std::make_tuple(10, std::make_tuple(3.14, "hello"));

(3)使用结构化绑定处理嵌套 tuple可以方便地处理嵌套的 tuple,通过一次性的声明对嵌套 tuple 进行解构和赋值:

auto [intValue, nestedTupleValue] = nestedTuple;
auto [doubleValue, stringValue] = nestedTupleValue;

(4)嵌套 tuple 的更复杂组合:多次的 tuple_cat 操作将多个 tuple 进行更复杂的组合,实现更为复杂的数据结构:

auto tuple1 = std::make_tuple(1, 2, 3);
auto tuple2 = std::make_tuple(4, 5, 6);
auto tuple3 = std::make_tuple(7, 8, 9);auto combinedTuple = std::tuple_cat(tuple1, tuple2, tuple3);
// combinedTuple 的类型为 std::tuple<int, int, int, int, int, int, int, int, int>

特别是结合结构化绑定的使用,可以简化对嵌套和组合 tuple 的操作,提高代码的可读性和可维护性。

四、tuple实例解析

(1)实例一:使用tuple进行函数返回多个值

#include <iostream>
#include <tuple>// 使用 std::make_tuple 返回多个值
std::tuple<int, double, float> calculateValues(int a, int b) {int sum = a + b;double difference = a - b;float product = a * b;float quotient = static_cast<float>(a) / b;return std::make_tuple(sum, difference, product * quotient);
}int main() {int x = 10, y = 5;// 使用 std::make_tuple 方式创建 tupleauto result1 = calculateValues(x, y);// 使用结构化绑定和 std::tie 方式解构 tuple 结果int s1, d1;float pq1;std::tie(s1, d1, pq1) = result1;std::cout << "Sum: " << s1 << ", Difference: " << d1 << ", Product*Quotient: " << pq1 << std::endl;// 使用直接通过 std::tuple 的构造函数方式创建 tuplestd::tuple<int, double, float> result2(x + y, x - y, x * y / static_cast<float>(x));int s2, d2;float pq2;std::tie(s2, d2, pq2) = result2;std::cout << "Sum: " << s2 << ", Difference: " << d2 << ", Product*Quotient: " << pq2 << std::endl;return 0;
}

函数接受两个整数参数,并返回了一个包含三个值的 tuple。在 main 函数中通过不同方式创建了 tuple 并使用结构化绑定解构 tuple 中的值。

(2)实例二:使用tuple进行数据结构的扩展

#include <iostream>
#include <tuple>int main() {// 创建一个包含不同类型的元素的 tuplestd::tuple<int, double> data1 = std::make_tuple(10, 3.14);// 定义结构体struct Person {std::string name;int age;};// 创建一个包含结构体和其他类型的元素的 tuplestd::tuple<Person, std::string, int> data2 = std::make_tuple(Person{"Alice", 30}, "Hello", 99);// 创建一个包含 tuple 的 tuplestd::tuple<int, std::tuple<double, std::string>> data3 = std::make_tuple(5, std::make_tuple(3.14, "pi"));// 创建一个更复杂的嵌套结构std::tuple<std::string, std::tuple<int, double>, std::tuple<float, char>> data4 = std::make_tuple("Nested", std::make_tuple(10, 3.14), std::make_tuple(4.5f, 'A'));// 访问和打印 tuple 中的元素std::cout << "Data 1: " << std::get<0>(data1) << ", " << std::get<1>(data1) << std::endl;std::cout << "Data 2: " << std::get<0>(std::get<0>(data2)).name << ", " << std::get<0>(std::get<0>(data2)).age << ", " << std::get<1>(data2) << ", " << std::get<2>(data2) << std::endl;std::cout << "Data 3: " << std::get<0>(data3) << ", " << std::get<0>(std::get<1>(data3)) << ", " << std::get<1>(std::get<1>(data3)) << std::endl;std::cout << "Data 4: " << std::get<0>(data4) << ", " << std::get<0>(std::get<1>(data4)) << ", " << std::get<1>(std::get<1>(data4)) << ", " << std::get<0>(std::get<2>(data4)) << ", " << std::get<1>(std::get<2>(data4)) << std::endl;return 0;
}

示例中创建了多个不同的 tuple 数据结构,每个都展示了一种常见的嵌套和组合方式。

(3)实例三:作为STL容器类的元素类型。例如 vector 和 map,使用 tuple 来存储不同类型的元素:

#include <iostream>
#include <vector>
#include <map>
#include <tuple>int main() {// 使用 tuple 作为 vector 的元素类型std::vector<std::tuple<int, double, std::string>> vec;// 添加元素到 vectorvec.push_back(std::make_tuple(10, 3.14, "Hello"));vec.push_back(std::make_tuple(20, 6.28, "World"));// 遍历 vector 中的元组for (const auto& item : vec) {std::cout << "Int value: " << std::get<0>(item) << std::endl;std::cout << "Double value: " << std::get<1>(item) << std::endl;std::cout << "String value: " << std::get<2>(item) << std::endl;}// 使用 tuple 作为 map 的 value 类型std::map<int, std::tuple<std::string, double>> myMap;// 添加元素到 mapmyMap[1] = std::make_tuple("John", 175.5);myMap[2] = std::make_tuple("Alice", 160.3);// 遍历 map 中的元组for (const auto& pair : myMap) {std::cout << "Key: " << pair.first << std::endl;std::cout << "Name: " << std::get<0>(pair.second) << std::endl;std::cout << "Height: " << std::get<1>(pair.second) << std::endl;}return 0;
}

五、tuple的性能和适用场景分析

tuple的性能
(1)与自定义结构体的性能比较:tuple不一定总是比自定义结构体优秀,tuple 也可能比使用自定义结构体更加低效,因为 tuple 通常需要进行运行时的动态内存分配。而使用自定义结构体会更加高效,因为它的内存布局更为紧凑,没有额外的开销。

(2)当tuple中包含的元素数量较少时,tuple 的性能通常是可以接受的,并且不会造成太大的性能损耗。

(3)当tuple中包含大量的元素时,tuple 的性能可能会受到影响,特别是在元组的创建、销毁和访问操作方面。

建议:如果对性能有较高要求,使用结构体或类来代替 tuple,因为结构体或类通常会在编译时进行优化,并且在内存布局方面更加紧凑。

tuple的适用场景

(1)返回多个值:当需要从函数中返回多个值时,使用 tuple 是一种方便的方式。

(2)想将多个数据项打包在一起,并且数据项的类型各不相同 时使用 tuple 可以很好地解决这个问题,避免创建新的自定义结构体。

(3)函数参数传递:函数需要接受多个不同类型的参数时,使用 tuple 可以很方便地将这些参数打包在一起,并传递给函数。

(4)中间结果存储:在一些算法或计算过程中会产生中间结果,使用 tuple 可以方便地将这些中间结果存储起来,以便之后使用。

(5)与算法库结合:在使用标准库中的算法时,tuple 也有其用武之地,比如可以方便地将多个数据项打包在一起,用作算法的输入或输出;或者遍历容器时使用tuple结构化绑定。

tuple与其他数据结构的对比

(1)Array(数组):数组是一种包含相同类型元素的数据结构,而 tuple 允许存储不同类型的元素。因此,如果要存储相同类型的元素,选择使用数组;而如果要存储不同类型的元素,使用 tuple。

(2)Pair(对):pair 是一种特殊的 tuple,它只能存储两个元素。因此,如果只需要存储两个元素,使用 pair;要存储三个及以上的元素,使用 tuple。

(3)Struct(结构体):结构体是一种自定义的数据结构,可以包含不同类型的数据成员,并且可以定义自己的成员函数。相比之下,tuple 是一个泛化的数据结构,通常用于临时存储一些松散相关的数据,而不需要为每个元素定义一个具体的名称。

(4)Vector(向量):vector 是一种动态数组,用于存储相同类型的元素。提供了动态扩展和访问元素的能力。tuple 则更适合用于存储固定数量的松散相关的数据,不要求动态操作。

总结

tuple 具有多种强大功能和灵活应用:

  1. 存储多个不同类型的值,便于打包和传递多元数据。
  2. 在函数中方便地返回多个值,简化了函数返回值的处理。
  3. 可以作为函数的参数类型,简化了参数列表的处理。
  4. 可以用于元编程和函数式编程范式,用于表示元组、模式匹配等。
  5. 与标准库中的算法结合使用,方便地作为算法的输入或输出。
  6. 可以用于解耦数据,不需要创建新的自定义数据结构。
  7. 使代码更简洁、更易读、更灵活。

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

继续理解Nacos的CP和AP架构模型!

本篇文章延续文章“如何理解Nacos册CP和AP架构模型”&#xff0c;大家可以配套一起学习。 Nacos注册中心处理HTTP注册请求 在文章“如何理解Nacos册CP和AP架构模型”中已经提到过&#xff0c;Nacos注册中心用Restful API InstanceController的方法register()处理HTTP类型的注…

21.串的处理

题目 import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner sc new Scanner(System.in);String str sc.nextLine();char[] c str.toCharArray();int n c.length;StringBuilder st new StringBuilder();int i 0;while(i<n)…

Python如何实现微信支付功能代码示例

微信支付是一种基于互联网的移动支付服务&#xff0c;由中国的即时通讯工具微信提供。用户可以通过微信支付在微信平台上进行在线支付、转账和收款。微信支付支持多种支付方式&#xff0c;包括银行卡支付、微信钱包余额支付、扫码支付等。用户可以用微信支付购买商品、支付账单…

吴恩达倾情推荐!28张图全解深度学习知识!

本文约7500字&#xff0c;建议阅读15分钟本文将从深度学习基础&#xff08;01-13&#xff09;、卷积网络&#xff08;14-22&#xff09;和循环网络&#xff08;23-28&#xff09;三个方面介绍该笔记。 吴恩达在推特上展示了一份由 TessFerrandez 完成的深度学习专项课程图&…

软件工程概论------文件管理

目录 1.文件的相关概念 2.文件目录 3.位示图 4.索引文件 5.例题 1.文件的相关概念 文件:具有符号名的、在逻辑上具有完整意义的一组相关信息项的集合。 逻辑结构:有结构的记录式文件、无结构的流式文件。 物理结构: 连续结构、链接结构、索引结构、多个物理块的索引表。 …

Spark调优解析-spark数据倾斜优化2(七)

1 数据倾斜优化 1.1为何要处理数据倾斜&#xff08;Data Skew&#xff09; 什么是数据倾斜 对Spark/Hadoop这样的大数据系统来讲&#xff0c;数据量大并不可怕&#xff0c;可怕的是数据倾斜。 何谓数据倾斜&#xff1f;数据倾斜指的是&#xff0c;并行处理的数据集中&#xf…

LeetCode做题总结 15. 三数之和、18. 四数之和 (Java)

不会做&#xff0c;参考了代码随想录和力扣官方题解&#xff0c;对此题进行整理。 X数之和 15. 三数之和代码思路20240103重写错误1错误2Java语言点总结 18. 四数之和代码思路20240104&#xff08;伪&#xff09;错误1 第一次剪枝错误2 第二次剪枝错误3 溢出 15. 三数之和 代码…

uniapp 创建组件组件

组件&#xff1a;用于将某个功能的 HTML、CSS、JS 封装到一个文件中&#xff0c;提高代码的复用性和可维护性。 创建组件 一、在根目录中创建 components 文件夹&#xff0c;右键点击新建组件。 二、输入组件名称、选择默认模板、点击创建组件。 三、在组件中正常编写内容即可…

java基础之Java8新特性-Stream(流)

简介 流&#xff08;Stream&#xff09;是 Java 8 引入的一种处理集合数据的抽象概念&#xff0c;它提供了一种更简洁、更灵活的方式来操作和处理集合数据。流可以看作是一系列元素的管道&#xff0c;可以对这些元素进行筛选、转换、排序、归约等操作&#xff0c;实现各种数据…

Java爬虫之Jsoup

1.Jsoup相关概念 Jsoup很多概念和js类似&#xff0c;可参照对比理解 Document &#xff1a;文档对象。每份HTML页面都是一个文档对象&#xff0c;Document 是 jsoup 体系中最顶层的结构。 Element&#xff1a;元素对象。一个 Document 中可以着包含着多个 Element 对象&#…

分析C++软件问题的常用分析工具及案例集锦详解

目录 1、库依赖关系查看工具Dependency Walker 2、GDI对象查看工具GDIview 3、PE信息查看工具PeViewer/MiTeC EXE Explorer 4、进程信息查看工具Process Explorer 5、进程监控工具Process Monitor 6、API函数调用监测工具API Monitor C软件异常排查从入门到精通系列教程&…

SpringBoot 调用mybatis报错:Invalid bound statement (not found):

启动SpringBoot报错&#xff1a;Invalid bound statement (not found): 参考此文排查 命中了第6条 记录一手坑爹的Invalid bound statement (not found)&#xff08;六个方面&#xff09; mapper文件路径配置错误 订正以后 问题解决