std::tuple
tuple簡史
C++ Reference對(duì)tuple的解釋是“fixed-size collection of heterogeneous values”额划,也就是有固定長度的異構(gòu)數(shù)據(jù)的集合妙啃。每一個(gè)C++代碼仔都很熟悉的std::pair
就是一種tuple
。但是std::pair
只能容納兩個(gè)數(shù)據(jù)俊戳,而C++11標(biāo)準(zhǔn)庫中定義的tuple
可以容納任意多個(gè)揖赴、任意類型的數(shù)據(jù)。
tuple的用法
C++ 11標(biāo)準(zhǔn)庫中的tuple
是一個(gè)模板類抑胎,使用時(shí)需要包含頭文件<tuple>
:
#include <tuple>
using tuple_type = std::tuple<int, double, char>;
tuple_type t1(1, 2.0, 'a');
不過我們一般都用std::make_tuple
函數(shù)來創(chuàng)建一個(gè)tuple
燥滑,使用std::make_tuple
的好處是不需要指定tuple
參數(shù)的類型,編譯器會(huì)自己推斷:
#include <iostream>
#include <tuple>
auto t1 = std::make_tuple(1, 2.0, 'a');
std::cout << typeid(t1).name() << std::endl
可以使用std::get
函數(shù)取出tuple
中的數(shù)據(jù):
auto t = std::make_tuple(1, 2.0, 'a');
std::cout << std::get<0>(t) << ", " << std::get<1>(t) << ", " << std::get<2>(t) << std::endl; // 1, 2.0, a
C++ 11標(biāo)準(zhǔn)庫中還定義了一些輔助類阿逃,方便我們?nèi)〉靡粋€(gè)tuple
類的信息:
using tuple_type = std::tuple<int, double, char>;
// tuple_size: 在編譯期獲得tuple元素個(gè)數(shù)
cout << std::tuple_size<tuple_type>::value << endl; // 3
// tuple_element: 在編譯期獲得tuple的元素類型
cout << typeid(std::tuple_element<2, tuple_type>::type).name() << endl; // c
關(guān)于tuple
的用法就簡要介紹到這里铭拧,C++ Reference上有關(guān)于std::tuple
的詳細(xì)介紹赃蛛,感興趣的同學(xué)可以去看看。下面我們著重講一下tuple
的實(shí)現(xiàn)原理搀菩。
tuple的實(shí)現(xiàn)原理
如果你對(duì)boost::tuple
有所了解的話焊虏,應(yīng)該知道boost::tuple
是使用遞歸嵌套實(shí)現(xiàn)的,這也是大多數(shù)類庫--比如Loki和 MS VC--實(shí)現(xiàn)tuple
的方法秕磷。而libc++
另辟蹊徑诵闭,采用了多重繼承的手法實(shí)現(xiàn)。libc++
的tuple
的源代碼極其復(fù)雜澎嚣,大量使用了元編程技巧疏尿,如果我一行行解讀這些源代碼,那本章就會(huì)變成C++模板元編程入門易桃。為了讓你有繼續(xù)看下去的勇氣褥琐,我將libc++ tuple
的源代碼簡化,實(shí)現(xiàn)了一個(gè)極簡版tuple
晤郑,希望能幫助你理解tuple
的工作原理敌呈。
tuple_size
我們先從輔助類開始:
// forward declaration
template<class ...T> class tuple;
template<class ...T> class tuple_size;
// 針對(duì)tuple類型的特化
template<class ...T>
class tuple_size<tuple<T...> > : public std::integral_constant<size_t, sizeof...(T)> {};
這個(gè)比較好理解,如果tuple_size
作用于一個(gè)tuple
造寝,則tuple_size
的值就是sizeof...(T)
的值磕洪。所以你可以這樣寫:
cout << tuple_size<tuple<int, double, char> >::value << endl; // 3
tuple_types
下一個(gè)輔助類就是tuple_types
:
template<class ...T> struct tuple_types{};
template<class T, size_t End = tuple_size<T>::value, size_t Start = 0>
struct make_tuple_types {};
template<class ...T, size_t End>
struct make_tuple_types<tuple<T...>, End, 0> {
typedef tuple_types<T...> type;
};
template<class ...T, size_t End>
struct make_tuple_types<tuple_types<T...>, End, 0> {
typedef tuple_types<T...> type;
};
這個(gè)簡化版的typle_types
并不做具體的事,就是純粹的類型定義诫龙。需要說明的是析显,如果你要使用這個(gè)簡化版的tuple_types
,最好保證End == sizeof...(T) - 1
签赃,否則有可能編譯器會(huì)報(bào)錯(cuò)谷异。
type_indices
下面這個(gè)有點(diǎn)復(fù)雜:
template<size_t ...value> struct tuple_indices {};
template<class IndexType, IndexType ...values>
struct integer_sequence {
template<size_t Start>
using to_tuple_indices = tuple_indices<(values + Start)...>;
};
template<size_t End, size_t Start>
using make_indices_imp = typename __make_integer_seq<integer_sequence, size_t, End - Start>::template to_tuple_indices<Start>;
template<size_t End, size_t Start = 0>
struct make_tuple_indices {
typedef make_indices_imp<End, Start> type;
};
__make_integer_seq
是LLVM編譯器的一個(gè)內(nèi)置的函數(shù),它的作用--顧名思義--是在編譯期生成一個(gè)序列锦聊,如果你寫下這樣的代碼:
__mkae_integer_seq<integer_sequence, size_t, 3>
則編譯器會(huì)將它展開成:
integer_sequence<0>, integer_sequence<1>, integer_sequence<2>
所以歹嘹,對(duì)于下面的代碼:
make_tuple_indices<3>
編譯器最終會(huì)展開成:
tuple_indices<0>, tuple_indices<1>, tuple_indices<2>
這樣就定義了一個(gè)tuple
的索引。
tuple_element
最后一個(gè)輔助類是tuple_element
:
namespace indexer_detail {
template<size_t Index, class T>
struct indexed {
using type = T;
};
template<class Types, class Indexes> struct indexer;
template<class ...Types, size_t ...Index>
struct indexer<tuple_types<Types...>, tuple_indices<Index...> > : public indexed<Index, Types>... {};
template<size_t Index, class T>
indexed<Index, T> at_index(indexed<Index, T> const&);
} // namespace indexer_detail
template<size_t Index, class ...Types>
using type_pack_element = typename decltype(indexer_detail::at_index<Index>(
indexer_detail::indexer<tuple_types<Types...>,
typename make_tuple_indices<sizeof...(Types)>::type>{}))::type;
template<size_t Index, class ...T>
struct tuple_element<Index, tuple_types<T...> > {
typedef type_pack_element<Index, T...> type;
};
template<size_t Index, class ...T>
struct tuple_element<Index, tuple<T...> > {
typedef type_pack_element<Index, T...> type;
};
我知道上面的代碼又讓你頭暈?zāi)垦?淄ィ晕視?huì)詳細(xì)解釋一下尺上。如果你寫下這樣的代碼:
tuple_element<1, tuple<int, double, char> >::type
編譯器會(huì)展開成(省略那些煩人的namespace限定符后):tuple_pack_element<1, int, double, char>
,進(jìn)而展開成
decltype(
at_index<1>(indexer<tuple_types<int, double, char>, tuple_indices<3>>{})
)
注意史飞,上面的代碼中定義了類indexer
作為函數(shù)at_index
的參數(shù)尖昏,而函數(shù)at_index
只接受at_index
類型的參數(shù),于是編譯器會(huì)來個(gè)向上轉(zhuǎn)型构资,將indexer
向上轉(zhuǎn)型成indexed<1,double>
(仔細(xì)想想為什么抽诉?),而indexed<1, double>::type
就是double
吐绵。
看似很復(fù)雜迹淌,其實(shí)無非就是文字代換而已河绽。
tuple
好了,酒水備齊了唉窃,下面上主菜:
template<size_t Index, class Head>
class tuple_leaf {
Head value;
public:
tuple_leaf() : vlaue(){}
template<class T>
explicit tuple_leaf(cosnt T& t) : value(t){}
Head& get(){return value;}
const Head& get() const {return value;}
};
tuple_leaf
是tuple
的基本組成單位耙饰,每一個(gè)tuple_leaf
都保存了一個(gè)索引(就是第一個(gè)模板參數(shù)),同時(shí)還有值纹份。
繼續(xù)看:
template<class Index, class ...T> struct tuple_imp;
template<size_t ...Index, class ...T>
struct tuple_imp<tuple_indices<Index...>, T...> :
public tuple_leaf<Index, T>... {
tuple_imp(){}
template<size_t ...Uf, class ...Tf, class ...U>
tuple_imp(tuple_indices<Uf...>, tuple_types<Tf...>, U&& ...u)
: tuple_leaf<Uf, Tf>(std::forward<U>(u))... {}
};
template<class ...T>
struct tuple {
typedef tuple_imp<typename make_tuple_indices<sizeof...(T)>::type, T...> base;
base base_;
tuple(const T& ...t)
: base(typename make_tuple_indices<sizeof...(T)>::type(),
typename make_tuple_types<tuple, sizeof...(T)>::type(),
t...){}
};
看到了吧苟跪,每一個(gè)tuple
都繼承自數(shù)個(gè)tuple_leaf
。而前面說過蔓涧,每個(gè)tuple_leaf
都有索引和值件已,所以定義一個(gè)tuple
所需要的信息都保存在這些tuple_leaf
中。如果有這樣的代碼
tuple(1, 2.0, 'a')
編譯器會(huì)展開成
struct tuple_imp : public tuple_leaf<0, int>, // value = 1
public tuple_leaf<1, double> // value = 2.0
public tuple_leaf<2, char> // value = 'a'
是不是有種腦洞大開的感覺元暴?
make_tuple 和 get
為了方便使用篷扩,標(biāo)準(zhǔn)庫還定義了函數(shù)make_tuple
和get
// make_tuple
template<class T>
struct make_tuple_return_imp {
typedef T type;
};
template<class T>
struct make_tuple_return {
typedef typename make_tuple_return_imp<typename std::decay<T>::type>::type type;
};
template<class ...T>
inline tuple<typename make_tuple_return<T>::type...> make_tuple<T&& ...t) {
return tuple<typename make_tuple_return<T>::type...>(std::forward<T>(t)...);
}
// get
template<size_t Index, class ...T>
inline typename tuple_element<Index, tuple<T...> >::type& get(tuple<T...>& t) {
typedef typename tuple_element<Index, tuple<T...> >::type type;
return static_cast<tuple_leaf<Index, type>&>(t.base_).get();
這些代碼我就不解釋了,留給你自己消化茉盏。
總結(jié)
本章展示的tuple
只是個(gè)簡化版的示例而已鉴未,要實(shí)現(xiàn)工業(yè)強(qiáng)度的tuple
,要做的工作還很多鸠姨。有興趣的同學(xué)可以去看看libc++
的源代碼铜秆。