当前位置:   article > 正文

C++-std:tuple元组的基本用法_std::make_tuple

std::make_tuple

一、元组简介

tuple是一个固定大小的不同类型值的集合,是泛化的std::pair。我们也可以把他当做一个通用的结构体来用,不需要创建结构体又获取结构体的特征,在某些情况下可以取代结构体使程序更简洁,直观。std::tuple理论上可以有无数个任意类型的成员变量,而std::pair只能是2个成员,因此在需要保存3个及以上的数据时就需要使用tuple元组了。

tuple(元组)在c++11中开始引用的。tuple看似简单,其实它是简约而不简单,可以说它是c++11中一个既简单又复杂的东东,关于它简单的一面是它很容易使用,复杂的一面是它内部隐藏了太多细节,要揭开它神秘的面纱时又比较困难。

二、tuple的创建和初始化

  1. std::tuple<T1, T2, TN> t1; //创建一个空的tuple对象(使用默认构造),它对应的元素分别是T1和T2...Tn类型,采用值初始化。
  2. std::tuple<T1, T2, TN> t2(v1, v2, ... TN); //创建一个tuple对象,它的元素分别是T1和T2 ...Tn类型; 要获取元素的值需要通过tuple的成员get<Ith>(obj)进行获取(Ith是指获取在tuple中的第几个元素,请看后面具体实例)。
  3. std::tuple<T1&> t3(ref&); // tuple的元素类型可以是一个引用
  4. std::make_tuple(v1, v2); // 像pair一样也可以通过make_tuple进行创建一个tuple对象

tuple的元素类型为引用:

  1. std::string name;
  2. std::tuple<string &, int> tpRef(name, 30);
  3. // 对tpRef第一个元素赋值,同时name也被赋值 - 引用
  4. std::get<0>(tpRef) = "Sven";
  5. // name输出也是Sven
  6. std::cout << "name: " << name << '\n';

三、有关tuple元素的操作

1、等价结构体

开篇讲过在某些时候tuple可以等同于结构体一样使用,这样既方便又快捷。如:

  1. struct person {
  2. char *m_name;
  3. char *m_addr;
  4. int *m_ages;
  5. };
  6. //可以用tuple来表示这样的一个结构类型,作用是一样的。
  7. std::tuple<const char *, const char *, int>

2、如何获取tuple元素个数

当有一个tuple对象但不知道有多少元素可以通过如下查询:

  1. // tuple_size
  2. #include <iostream> // std::cout
  3. #include <tuple> // std::tuple, std::tuple_size
  4. int main ()
  5. {
  6. std::tuple<int, char, double> mytuple (10, 'a', 3.14);
  7. std::cout << "mytuple has ";
  8. std::cout << std::tuple_size<decltype(mytuple)>::value;
  9. std::cout << " elements." << '\n';
  10. return 0;
  11. }
  12. //输出结果:
  13. mytuple has 3 elements

3、如何获取元素的值

获取tuple对象元素的值可以通过get<Ith>(obj)方法进行获取;

  • Ith - 是想获取的元素在tuple对象中的位置。
  • obj - 是想获取tuple的对象
  1. // tuple_size
  2. #include <iostream> // std::cout
  3. #include <tuple> // std::tuple, std::tuple_size
  4. int main ()
  5. {
  6. std::tuple<int, char, double> mytuple (10, 'a', 3.14);
  7. std::cout << "mytuple has ";
  8. std::cout << std::tuple_size<decltype(mytuple)>::value;
  9. std::cout << " elements." << '\n';
  10. //获取元素
  11. std::cout << "the elements is: ";
  12. std::cout << std::get<0>(mytuple) << " ";
  13. std::cout << std::get<1>(mytuple) << " ";
  14. std::cout << std::get<2>(mytuple) << " ";
  15. std::cout << '\n';
  16. return 0;
  17. }
  18. //输出结果:
  19. mytuple has 3 elements.
  20. the elements is: 10 a 3.14

tuple不支持迭代,只能通过元素索引(或tie解包)进行获取元素的值。

但是给定的索引必须是在编译器就已经给定,不能在运行期进行动态传递,否则将发生编译错误:

  1. for(int i=0; i<3; i++)
  2. std::cout << std::get<i>(mytuple) << " "; //将引发编译错误

4、获取元素的类型

 要想得到元素类型可以通过tuple_element方法获取,如有以下元组对象:

  1. std::tuple<std::string, int> tp("Sven", 20);
  2. // 得到第二个元素类型
  3. std::tuple_element<1, decltype(tp)>::type ages; // ages就为int类型
  4. ages = std::get<1>(tp);
  5. std::cout << "ages: " << ages << '\n';
  6. //输出结果:
  7. ages: 20

5、利用tie进行解包元素的值

如同pair一样也是可以通过tie进行解包tuple的各个元素的值。如下tuple对象有4个元素,通过tie解包将会把这4个元素的值分别赋值给tie提供的4个变量中。

  1. #include <iostream>
  2. #include <tuple>
  3. #include <utility>
  4. int main(int argc, char **argv) {
  5. std::tuple<std::string, int, std::string, int> tp;
  6. tp = std::make_tuple("Sven", 25, "Shanghai", 21);
  7. // 定义接收变量
  8. std::string name;
  9. std::string addr;
  10. int ages;
  11. int areaCode;
  12. std::tie(name, ages, addr, areaCode) = tp;
  13. std::cout << "Output: " << '\n';
  14. std::cout << "name: " << name <<", ";
  15. std::cout << "addr: " << addr << ", ";
  16. std::cout << "ages: " << ages << ", ";
  17. std::cout << "areaCode: " << areaCode << '\n';
  18. return 0;
  19. }
  20. //输出结果:
  21. Output:
  22. name: Sven, addr: Shanghai, ages: 25, areaCode: 21

但有时候tuple包含的多个元素时只需要其中的一个或两个元素,如此可以通过std::ignore进行变量占位,这样将会忽略提取对应的元素。可以修改上述例程:

  1. #include <iostream>
  2. #include <tuple>
  3. #include <utility>
  4. int main(int argc, char **argv) {
  5. std::tuple<std::string, int, std::string, int> tp;
  6. tp = std::make_tuple("Sven", 25, "Shanghai", 21);
  7. // 定义接收变量
  8. std::string name;
  9. std::string addr;
  10. int ages;
  11. int areaCode = 110;
  12. std::tie(name, ages, std::ignore, std::ignore) = tp;
  13. std::cout << "Output: " << '\n';
  14. std::cout << "name: " << name <<", ";
  15. std::cout << "addr: " << addr << ", ";
  16. std::cout << "ages: " << ages << ", ";
  17. std::cout << "areaCode: " << areaCode << '\n';
  18. return 0;
  19. }
  20. //输出结果:
  21. Output:
  22. name: Sven, addr: , ages: 25, areaCode: 110

6、tuple元素的引用

前面已经列举了将引用作为tuple的元素类型。下面通过引用搭配make_tuple()可以提取tuple的元素值,将某些变量值设给它们,并通过改变这些变量来改变tuple元素的值:

  1. #include <iostream>
  2. #include <tuple>
  3. #include <functional>
  4. int main(int argc, char **agrv) {
  5. std::tuple<std::string, int, float> tp1("Sven Cheng", 77, 66.1);
  6. std::string name;
  7. int weight;
  8. float f;
  9. auto tp2 = std::make_tuple(std::ref(name), std::ref(weight), std::ref(f)) = tp1;
  10. std::cout << "Before change: " << '\n';
  11. std::cout << "name: " << name << ", ";
  12. std::cout << "weight: " << weight << ", ";
  13. std::cout << "f: " << f << '\n';
  14. name = "Sven";
  15. weight = 80;
  16. f = 3.14;
  17. std::cout << "After change: " << '\n';
  18. std::cout << "element 1st: " << std::get<0>(tp2) << ", ";
  19. std::cout << "element 2nd: " << std::get<1>(tp2) << ", ";
  20. std::cout << "element 3rd: " << std::get<2>(tp2) << '\n';
  21. return 0;
  22. }
  23. //输出结果:
  24. Before change:
  25. name: Sven Cheng, weight: 77, f: 66.1
  26. After change:
  27. element 1st: Sven, element 2nd: 80, element 3rd: 3.14

7、tuple交换

  1. tuple<int, std::string, float> t4(11, "Test", 3.14);
  2. cout << get<0>(t3) << " " << get<0>(t4) << endl;
  3. t3.swap(t4);
  4. cout << get<0>(t3) << " " << get<0>(t4) << endl;

8、排序

  1. bool cmp(tuple<string,int,char> a,tuple<string,int,char> b){
  2.  return a<b;
  3.  return get<1>(a)<get<1>(b); // 也可以按某列排序
  4. }
  5. main(){
  6.  tuple<string,int,char> my_tuple[10];
  7.  my_tuple[0] = std::make_tuple ("Pipr",42,'a');
  8.  my_tuple[1] = std::make_tuple ("Piper",41,'a');
  9.  my_tuple[2] = std::make_tuple ("Pper",45,'a');
  10.  my_tuple[3] = std::make_tuple ("Pier",49,'a');
  11.  for(int i=0;i<4;++i){
  12.   cout << get<0>(my_tuple[i]) << " " << get<1>(my_tuple[i]) << " ";
  13.  }
  14.  cout << endl;
  15.  sort(my_tuple,my_tuple+4,cmp);
  16.  for(int i=0;i<4;++i){
  17.   cout << get<0>(my_tuple[i]) << " " << get<1>(my_tuple[i]) << " ";
  18.  }
  19.  cout << endl;
  20. }

C++ tuple元组的基本用法(总结)_sevencheng798的博客-CSDN博客_c++ tuple赋值

C++ tuple元组的基本用法(总结)_C 语言_脚本之家

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/笔触狂放9/article/detail/305956
推荐阅读
相关标签
  

闽ICP备14008679号