tuple用法简介
这次要讲的内容是:c++11中的tuple(元组)。tuple看似简单,其实它是简约而不简单,可以说它是c++11中一个既简单又复杂的东东,关于它简单的一面是它很容易使用,复杂的一面是它内部隐藏了太多细节,要揭开它神秘的面纱时又比较困难。
tuple是一个固定大小的不同类型值的集合,是泛化的std::pair。和c#中的tuple类似,但是比c#中的tuple强大得多。我们也可以把他当做一个通用的结构体来用,不需要创建结构体又获取结构体的特征,在某些情况下可以取代结构体使程序更简洁,直观。
基本用法
注意包含头文件<tuple>
构造一个tuple
tuple<const char*, int>tp = make_tuple(sendPack,nSendSize); //构造一个tuple
这个tuple等价于一个结构体
struct A
{
char* p;
int len;
};
用tuple<const char*, int>tp
就可以不用创建这个结构体了,而作用是一样的,是不是更简洁直观了。还有一种方法也可以创建元组,用std::tie
,它会创建一个元组的左值引用。
auto tp = return std::tie(1, "aa", 2);
//tp的类型实际是:
std::tuple<int&,string&, int&>
再看看如何获取它的值:
const char* data = tp.get<0>(); //获取第一个值
int len = tp.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::forward_as_tuple(10, std::string(20, 'a')));
它实际上创建了一个类似于std::tuple<int&&, std::string&&>
类型的tuple。
我们还可以通过tuple_cat连接多个tupe
#include <iostream>
#include <string>
#include <tuple>
#include <utility>
using namespace std;
template <class Tuple, size_t N>
struct print_imp {
static void print(ostream& out, const Tuple& t)
{
print_imp<Tuple, N - 1>::print(out, t); //1
out << ", " << get<N - 1>(t); //2
}
};
template <class Tuple>
struct print_imp<Tuple, 1> {
static void print(ostream& out, const Tuple& t)
{
out << get<0>(t);
}
};
template <class... Args>
ostream& operator<<(ostream& out, const tuple<Args...>& t)
{
out << '(';
print_imp<decltype(t), sizeof...(Args)>::print(out, t);
return out << ')';
}
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;
cout << t2 << endl;
}
输出结果:
(10, Test, 3.14, Foo, bar, 10, Test, 3.14, 10)
到这里tuple的用法介绍完了,是不是很简单,也很容易使用,相信你使用它之后就离不开它了。