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

  tuple是一个固定大小的不同类型值的集合,是泛化的std::pair。和c#中的tuple类似,但是比c#中的tuple强大得多。我们也可以把他当做一个通用的结构体来用,不需要创建结构体又获取结构体的特征,在某些情况下可以取代结构体使程序更简洁,直观。

基本用法

构造一个tuple

  1. tuple<const char*, int>tp = make_tuple(sendPack,nSendSize); //构造一个tuple

这个tuple等价于一个结构体

  1. struct A
  2. {
  3. char* p;
  4. int len;
  5. };

用tuple<const char*, int>tp就可以不用创建这个结构体了,而作用是一样的,是不是更简洁直观了。还有一种方法也可以创建元组,用std::tie,它会创建一个元组的左值引用。

  1. auto tp = return std::tie(1, "aa", 2);
  2. //tp的类型实际是:
  3. std::tuple<int&,string&, int&>

再看看如何获取它的值:

  1. const char* data = tp.get<0>(); //获取第一个值
  2. int len = tp.get<1>(); //获取第二个值

还有一种方法也可以获取元组的值,通过std::tie解包tuple

  1. int x,y;
  2. string a;
  3. std::tie(x,a,y) = tp;

通过tie解包后,tp中三个值会自动赋值给三个变量。

解包时,我们如果只想解某个位置的值时,可以用std::ignore占位符来表示不解某个位置的值。比如我们只想解第三个值时:

  1. std::tie(std::ignore,std::ignore,y) = tp; //只解第三个值了

还有一个创建右值的引用元组方法:forward_as_tuple。

  1. std::map<int, std::string> m;
  2. m.emplace(std::forward_as_tuple(10, std::string(20, 'a')));

它实际上创建了一个类似于std::tuple<int&&, std::string&&>类型的tuple。

我们还可以通过tuple_cat连接多个tupe

  1. int main()
  2. {
  3. std::tuple<int, std::string, float> t1(10, "Test",
  4. 3.14);
  5. int n = 7;
  6. auto t2 = std::tuple_cat(t1, std::make_pair("Foo",
  7. "bar"), t1, std::tie(n));
  8. n = 10;
  9. print(t2);
  10. }

输出结果:

  1. (10, Test, 3.14, Foo, bar, 10, Test, 3.14, 10)

  到这里tuple的用法介绍完了,是不是很简单,也很容易使用,相信你使用它之后就离不开它了。我前面说过tuple是简约而不简单。它有很多高级的用法。它和模板元关系密切,要介绍它的高级用法的时候,读者需要一定的模板元基础,如果你只是把它当一个泛型的pair去使用时,这部分可以不看,如果你想用它高级用法的时候就往下看。让我们要慢慢揭开tuple神秘的面纱。

tuple的高级用法

获取tuple中某个位置元素的类型

  通过std::tuple_element获取元素类型。

  1. template<typename Tuple>
  2. void Fun(Tuple& tp)
  3. {
  4. std::tuple_element<0,Tuple>::type first = std::get<0>
  5. (mytuple);
  6. std::tuple_element<1,Tuple>::type second = std::get<1>
  7. (mytuple);
  8. }

  

获取tuple中元素的个数:

tuple t;

int size = std::tuple_size<decltype(t))>::value;

遍历tuple中的每个元素

  因为tuple的参数是变长的,也没有for_each函数,如果我们想遍历tuple中的每个元素,需要自己写代码实现。比如我要打印tuple中的每个元素。

  1. template<class Tuple, std::size_t N>
  2. struct TuplePrinter {
  3. static void print(const Tuple& t)
  4. {
  5. TuplePrinter<Tuple, N - 1>::print(t);
  6. std::cout << ", " << std::get<N - 1>(t);
  7. }
  8. };
  9.  
  10. template<class Tuple>
  11. struct TuplePrinter<Tuple, 1>{
  12. static void print(const Tuple& t)
  13. {
  14. std::cout << std::get<0>(t);
  15. }
  16. };
  17.  
  18. template<class... Args>
  19. void PrintTuple(const std::tuple<Args...>& t)
  20. {
  21. std::cout << "(";
  22. TuplePrinter<decltype(t), sizeof...(Args)>::print(t);
  23. std::cout << ")\n";
  24. }

根据tuple元素值获取其对应的索引位置

  1. namespace detail
  2. {
  3. template<int I, typename T, typename... Args>
  4. struct find_index
  5. {
  6. static int call(std::tuple<Args...> const& t, T&& val)
  7. {
  8. return (std::get<I - 1>(t) == val) ? I - 1 :
  9. find_index<I - 1, T, Args...>::call(t, std::forward<T>(val));
  10. }
  11. };
  12.  
  13. template<typename T, typename... Args>
  14. struct find_index<0, T, Args...>
  15. {
  16. static int call(std::tuple<Args...> const& t, T&& val)
  17. {
  18. return (std::get<0>(t) == val) ? 0 : -1;
  19. }
  20. };
  21. }
  22.  
  23. template<typename T, typename... Args>
  24. int find_index(std::tuple<Args...> const& t, T&& val)
  25. {
  26. return detail::find_index<0, sizeof...(Args) - 1, T, Args...>::
  27. call(t, std::forward<T>(val));
  28. }
  29.  
  30. int main()
  31. {
  32. std::tuple<int, int, int, int> a(2, 3, 1, 4);
  33. std::cout << find_index(a, 1) << std::endl; // Prints 2
  34. std::cout << find_index(a, 2) << std::endl; // Prints 0
  35. std::cout << find_index(a, 5) << std::endl; // Prints -1 (not found)
  36. }

展开tuple,并将tuple元素作为函数的参数。这样就可以根据需要对tuple元素进行处理了

  1. #include <tuple>
  2. #include <type_traits>
  3. #include <utility>
  4.  
  5. template<size_t N>
  6. struct Apply {
  7. template<typename F, typename T, typename... A>
  8. static inline auto apply(F && f, T && t, A &&... a)
  9. -> decltype(Apply<N-1>::apply(
  10. ::std::forward<F>(f), ::std::forward<T>(t),
  11. ::std::get<N-1>(::std::forward<T>(t)),
  12. ::std::forward<A>(a)...
  13. ))
  14. {
  15. return Apply<N-1>::apply(::std::forward<F>(f),
  16. ::std::forward<T>(t),
  17. ::std::get<N-1>(::std::forward<T>(t)),
  18. ::std::forward<A>(a)...
  19. );
  20. }
  21. };
  22.  
  23. template<>
  24. struct Apply<0> {
  25. template<typename F, typename T, typename... A>
  26. static inline auto apply(F && f, T &&, A &&... a)
  27. -> decltype(::std::forward<F>(f)
  28. (::std::forward<A>(a)...))
  29. {
  30. return ::std::forward<F>(f)(::std::forward<A>
  31. (a)...);
  32. }
  33. };
  34.  
  35. template<typename F, typename T>
  36. inline auto apply(F && f, T && t)
  37. -> decltype(Apply< ::std::tuple_size<
  38. typename ::std::decay<T>::type
  39. >::value>::apply(::std::forward<F>(f),
  40. ::std::forward<T>(t)))
  41. {
  42. return Apply< ::std::tuple_size<
  43. typename ::std::decay<T>::type
  44. >::value>::apply(::std::forward<F>(f),
  45. ::std::forward<T>(t));
  46. }
  47.  
  48. void one(int i, double d)
  49. {
  50. std::cout << "function one(" << i << ", " << d <<
  51. ");\n";
  52. }
  53. int two(int i)
  54. {
  55. std::cout << "function two(" << i << ");\n";
  56. return i;
  57. }
  58.  
  59. //测试代码
  60. int main()
  61. {
  62. std::tuple<int, double> tup(23, 4.5);
  63. apply(one, tup);
  64.  
  65. int d = apply(two, std::make_tuple(2));
  66.  
  67. return 0;
  68. }

  看到这里,想必大家对tuple有了一个全面的认识了吧,怎么样,它是简约而不简单吧。对模板元不熟悉的童鞋可以不看tuple高级用法部分,不要为看不懂而捉急,没事的,高级部分一般用不到,知道基本用法就够用了。

tuple和vector比较:

vector只能容纳同一种类型的数据,tuple可以容纳任意类型的数据;

vector和variant比较:

二者都可以容纳不同类型的数据,但是variant的类型个数是固定的,而tuple的类型个数不是固定的,是变长的,更为强大。

tuple元组详解的更多相关文章

  1. Python之tuple元组详解

    元组:有序,一级元素不可以修改.不能被增加或删除(元组是可迭代对象) 一般写法括号内最后面加个英文逗号用来区分: test  =  (,) test1 = (11,22,) 例: test = (12 ...

  2. python的dict,set,list,tuple应用详解

    python的dict,set,list,tuple应用详解 本文深入剖析了python中dict,set,list,tuple应用及对应示例,有助于读者对其概念及原理的掌握.具体如下: 1.字典(d ...

  3. Tuple元组 、 ValueTuple 值元组详解

    Tuple元组 Tuple是C# 4.0时出的新特性,.Net Framework 4.0以上版本可用. 元组是一种数据结构,具有特定数量和元素序列,与数组不同,元祖中的元素可以不同的数据类型.比如设 ...

  4. Python元组详解

    元组的特征 元组类型的名字是tuple 元组的一级元素不可被修改.不能增加或者删除: 元组和列表的书写区别是将中括号改成了小括号: 为方便区分元组和普通方法的参数,一般在元组的最后一个元素后保持加一个 ...

  5. Python 元组(Tuple)操作详解

    Python的元组与列表类似,不同之处在于元组的元素不能修改,元组使用小括号, 列表使用方括号,元组创建很简单,只需要在括号中添加元素,并使用逗号隔开即可 一.创建元组 代码如下: tup1 = (' ...

  6. python 列表、元组 详解

    python中有6种序列的内置类型,分别为:列表,元组,字符串,Unicode字符串,buffer对象和xrange对象 列表和元组是最常见两种类型. 下面将以列表(list)和元组(tuple)为例 ...

  7. python数据类型及其详解

    一.数据类型介绍 1.字符串(string) 三种类型:单引号,双引号,三引号 a = 'jam' b = "JamHsiao" c = '''JAMHSIAO''' print( ...

  8. Python - 元组(tuple) 详解 及 代码

    元组(tuple) 详解 及 代码 本文地址: http://blog.csdn.net/caroline_wendy/article/details/17290967 元组是存放任意元素集合,不能修 ...

  9. 详解C# Tuple VS ValueTuple(元组类 VS 值元组)

    C# 7.0已经出来一段时间了,大家都知道新特性里面有个对元组的优化,并且网上也有大量的介绍,这里利用详尽的例子详解Tuple VS ValueTuple(元组类VS值元组),10分钟让你更了解Val ...

随机推荐

  1. js程序中美元符号$是什么(转)

    $符号在php中是表示变量的特征字符, 在js中它也有很多作用, 一般我们用来命名一个函数名称,获取id的1.首先可以用来表示变量, 比如变量 var s='asdsd'或var $s='asdasd ...

  2. Cocoa pod导入第三方框架遇到的那点事儿

    废话不多说,直接上干货. 以下所有操作均是在Cocoapod导入第三方库,并且是.xcworkspace的工程里面操作的, 1.导入头文件找不到,也就是所谓的,not find ''AFNetwork ...

  3. HTML的代码规范

    一.语法 用两个空格来代替制表符(tab) 2.嵌套元素应当缩进一次(即两个空格). 3.对于属性的定义,确保全部使用双引号,绝不要使用单引号. 4.不要省略可选的结束标签(例如,</li> ...

  4. C# 压缩图片到指定宽度,假如图片小于指定宽度 判断图片大小是否大于指定大小(KB) 如果大于则压缩图片质量 宽高不变

    class Program { static void Main(string[] args) {//G:\zhyue\backup\projects\Test\ConsoleApplication1 ...

  5. sublime text插件推荐

    time: 2016-05-23 13:00 --- 用了Sublime Text有一段时间了,之前被周围的几个前端小伙伴灌输的思想就是,牛b的前端程序员都是手写代码的,用代码提示啥的都low,然而上 ...

  6. windows域渗透实战

    测试环境 域控: 192.168.211.130 已经控制的机器: 192.168.211.133 获取网络信息 查看机器的网络信息 ipconfig /all # 查看 网卡信息,获取dns 服务器 ...

  7. Vue 框架-10-搭建脚手架 CLI

    Vue 框架-10-搭建脚手架 CLI + 批处理快捷启动 脚手架是通过 webpack 搭建的开发环境 使用 ES6 语法 打包和压缩 JS 为一个文件 项目文件在环境中,而不是浏览器 实现页面自动 ...

  8. Week5——applet

    1.定义 applet是一种Java程序.它一般运行在支持Java的Web浏览器内.因为它有完整的Java API支持,所以applet是一个全功能的Java应用程序. 2.特点(不同于Java  a ...

  9. zTree实现权限列表简单实例

    zTree的简单实例 zTree 是一个依靠jQuery 实现的多功能 "树插件".优异的性能.灵活的配置.多种功能的组合是 zTree 最大优点. zTree官网 zTreeAP ...

  10. CSS3 响应式web设计,CSS3 Media Queries

    两种方式,一种是直接在link中判断设备的尺寸,然后引用不同的css文件: <link rel="stylesheet" type="text/css" ...