原文 :https://www.cnblogs.com/qicosmos/p/3318070.html

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

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

基本用法

构造一个tuple

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

这个tuple等价于一个结构体

struct A
{
char* p;
int len;
};

View Code

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

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

再看看如何获取它的值:

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

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

int x,y;
string a;
std::tie(x,a,y) = tp; 

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

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

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

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

std::map<int, std::string> m;

m.emplace(std::piecewise_construct,
std::forward_as_tuple(10),
std::forward_as_tuple(20, 'a'));

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

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

int main()
{
std::tuple<int, std::string, float> t1(10, "Test",
3.14);
int n = 7;
auto t2 = std::tuple_cat(t1, std::make_pair("Foo",
"bar"), t1, std::tie(n));
n = 10;
print(t2);
}

View Code

输出结果:

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

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

tuple的高级用法

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

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

template<typename Tuple>
void Fun(Tuple& tp)
{
std::tuple_element<0,Tuple>::type first = std::get<0>
(mytuple);
std::tuple_element<1,Tuple>::type second = std::get<1>
(mytuple);
}

  

获取tuple中元素的个数:

tuple t;

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

遍历tuple中的每个元素

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

template<class Tuple, std::size_t N>
struct TuplePrinter {static void print(const Tuple& t){TuplePrinter<Tuple, N - 1>::print(t);std::cout << ", " << std::get<N - 1>(t);}
};template<class Tuple>
struct TuplePrinter<Tuple, 1>{static void print(const Tuple& t){std::cout << std::get<0>(t);}
};template<class... Args>
void PrintTuple(const std::tuple<Args...>& t)
{std::cout << "(";TuplePrinter<decltype(t), sizeof...(Args)>::print(t);std::cout << ")\n";
}

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

namespace detail
{template<int I, typename T, typename... Args>struct find_index{static int call(std::tuple<Args...> const& t, T&& val){return (std::get<I - 1>(t) == val) ? I - 1 :find_index<I - 1, T, Args...>::call(t, std::forward<T>(val));}};template<typename T, typename... Args>struct find_index<0, T, Args...>{static int call(std::tuple<Args...> const& t, T&& val){return (std::get<0>(t) == val) ? 0 : -1;}};
}template<typename T, typename... Args>
int find_index(std::tuple<Args...> const& t, T&& val)
{return detail::find_index<0, sizeof...(Args) - 1, T, Args...>::call(t, std::forward<T>(val));
}int main()
{std::tuple<int, int, int, int> a(2, 3, 1, 4);std::cout << find_index(a, 1) << std::endl; // Prints 2std::cout << find_index(a, 2) << std::endl; // Prints 0std::cout << find_index(a, 5) << std::endl; // Prints -1 (not found)
}

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

#include <tuple>
#include <type_traits>
#include <utility>template<size_t N>
struct Apply {
template<typename F, typename T, typename... A>
static inline auto apply(F && f, T && t, A &&... a)
-> decltype(Apply<N-1>::apply(
::std::forward<F>(f), ::std::forward<T>(t),
::std::get<N-1>(::std::forward<T>(t)),
::std::forward<A>(a)...
))
{
return Apply<N-1>::apply(::std::forward<F>(f),
::std::forward<T>(t),
::std::get<N-1>(::std::forward<T>(t)),
::std::forward<A>(a)...
);
}
};template<>
struct Apply<0> {
template<typename F, typename T, typename... A>
static inline auto apply(F && f, T &&, A &&... a)
-> decltype(::std::forward<F>(f)
(::std::forward<A>(a)...))
{
return ::std::forward<F>(f)(::std::forward<A>
(a)...);
}
};template<typename F, typename T>
inline auto apply(F && f, T && t)
-> decltype(Apply< ::std::tuple_size<
typename ::std::decay<T>::type
>::value>::apply(::std::forward<F>(f),
::std::forward<T>(t)))
{
return Apply< ::std::tuple_size<
typename ::std::decay<T>::type
>::value>::apply(::std::forward<F>(f),
::std::forward<T>(t));
}void one(int i, double d)
{
std::cout << "function one(" << i << ", " << d <<
");\n";
}
int two(int i)
{
std::cout << "function two(" << i << ");\n";
return i;
}//测试代码
int main()
{
std::tuple<int, double> tup(23, 4.5);
apply(one, tup);int d = apply(two, std::make_tuple(2));return 0;
}

View Code

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

tuple和vector比较:

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

tuple和variant比较:

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

2020-09-11 C++ 11 tuple元组和tie()的简单介绍相关推荐

  1. 京东:截至11月11日00:09 累计下单金额超2000亿元

    11月11日消息,根据京东官方公布的数据显示,2020年11月1日00:00至11月11日00:09,京东11.11全球热爱季累计下单金额已经突破2000亿元. 据悉,"2020京东11.1 ...

  2. c++11新特性--tuple

    tuple C++11 标准新引入了一种类模板,命名为 tuple(中文可直译为元组).tuple 最大的特点是:实例化的对象可以存储任意数量.任意类型的数据. 实例化 tuple 模板类对象常用的方 ...

  3. 共享可写节包含重定位_艾瑞咨询:2020年数说双11电商购物节报告

    回复"2020数说双11电商购物节"获取完整版报告 <2020年数说双11电商购物节报告>基于艾瑞十八年对互联网流量研究的研究经验.数据积累以及艾瑞数据监测工具多平台网 ...

  4. 2020年的双11,阿里需要什么样的渲染方案?

    阿里妹导读:前端技术的"新陈代谢"是有目共睹的,新技术的不断发展也推动着前端应用场景的不断扩大,从 Web .Weex 到 Node.js 再到 FaaS.我们在发展中看不变的部分 ...

  5. 2020年天猫双11:11日0点30分 实时成交额突破3723亿

    11月11日消息,根据天猫公布的数据显示,11月1日至11日0点30分,2020年天猫双11全球狂欢季实时成交额突破3723亿,实时成交额超过1亿元的品牌已经超过300个. 在2020天猫双11全球狂 ...

  6. 计算机中的配置是Intel,2020年4月11日,中高端电脑配置清单(INTEL和AMD)

    2020年4月11日(写个时间表明是这个时间的行情) 一步一步似魔鬼的步伐,终于来到了免费能算个大哥的配置了, 到这么个价位的配置,其实选择就非常多,特别是在外观上面有很多的选择,一个机箱可能有些小伙 ...

  7. 2020战地指挥官服务器维护,《神雕侠侣》2020年6月11日更新维护新服开启公告

    亲爱的玩家: 为了给您提供更好的游戏体验,我们将于2020年6月11日上午8:00-11:00对所有服务器进行停机更新维护,预计维护时间3小时. 若提前完成维护内容,我们将提前开服.对于停机期间给您带 ...

  8. 2020淘宝双11超级星秀猫怎么踢人 最强星秀猫怎么退出队伍

    2020年的天猫双十一的喵币活动是叫-超级星秀猫,还是一个养猫的活动,不过是可以组队的,有的朋友是想把临时的人踢出队伍或者是自己离开队伍,但是不知道怎么操作,下面就来为大家详细的介绍一下. 1.踢人: ...

  9. 2020年天猫双11官宣:分两波购买 比往年多3天

    10月20日消息,天猫正式发布新一代「天猫双11全球狂欢季」,代号:双节棍.此次升级最大的变化是:不只在11月11日一天爆发,消费者可分两波购买,11月1日-3日是第一波,11月11日为第二波. 这意 ...

最新文章

  1. cdh的集成phoenix安装_环境篇:Kylin3.0.1集成CDH6.2.0
  2. 突破极限 解决大硬盘上安装Sco Unix新思路
  3. 对静态区,栈,堆的理解
  4. javascript 函数声明与函数表达式的区别
  5. poj 3131 双向搜索+hash判重
  6. Python os.path模块的使用
  7. TCP/IP 三次握手和四次挥手
  8. 9.3 LSMW程序创建操作手册 第5 6 7步
  9. restful是什么
  10. Managed I/O Completion Ports (IOCP)
  11. 免费的配音软件,堪比魔云熙声音。
  12. java几何画板_geogebra几何画板下载
  13. 星际争霸2的移动机制-初探
  14. ubuntu 14.04.1 smbd环境搭建
  15. 巅峰对决:node.js和php性能测试
  16. 为什么你和同事关系处不好?
  17. mysql 的字符集、比较规则和服务端、客户端的相关配置
  18. 【Shotcut】用最短路径编辑一个视频
  19. TRC丨艾美捷TRC a,a-二甲基-4-说明书
  20. C程序设计复习(1)

热门文章

  1. 『影视评论』 [电影众论]越狱中觉得比较经典的十段台词(转载)
  2. 荣誉勋章空降神兵诺基亚Java_荣誉勋章
  3. mysql: navicat for mysql 执行快捷键
  4. Android Palette
  5. java中contextpath是什么,你如何从JavaScript获得contextPath,正确的方式?
  6. Unity Shader - Making multiple shader program variants 制作shader程序多变体
  7. 搭建python开发环境_花瓣网_陪你做生活的设计师(发现、采集你喜欢的灵感、家居、穿搭、婚礼、美食、旅行、美图、商品等)...
  8. 「组装式开发」企业数字化转型的及时雨
  9. Android中使用PhotoView和ViewPager查看图片,并保存到本地
  10. Android apk用pm命令安装apk碰到问题汇总