填写std :: tuple

时间:2013-12-28 16:28:51

标签: c++ tuples variadic-templates boost-fusion

我有一个重载函数,看起来像:

template<typename T>
T getColumn(size_t i);

template<>
std::string getColumn<std::string>(size_t i) {
    if(i == 0)
        return "first";
    else
        return "other";
}

template<>
int getColumn<int>(size_t i) {
    return i*10;
}

// ...

现在我想实现这个功能

template<typename... Values>
std::tuple<Values...> getColumns();

创建一个元组(用于返回值)并为元组的每个元素调用getColumn(保存该元素中的返回值),其中i是元素的位置。生成getColumn的返回值的代码被简化(实际上它从数据库中获取值)。

但我不知道该怎么做。

我最好的尝试是使用boost :: fusion :: for_each,但我无法将i交给getColumn

另一个尝试是使用boost :: fusion的迭代器,但这也没有用:

namespace fusion = boost::fusion;
tuple<Values...> t;
auto it = fusion::begin(t);
while(it != fusion::end(t)) {
    getColumn(distance(fusion::begin(t), it), fusion::deref(it));
    it = fusion::next(it); // error: assignment not allowed
}

如何为getColumn中的Values...的每个类型调用i std::tuple的正确值,并将结果保存在{{1}}?

2 个答案:

答案 0 :(得分:3)

您需要将参数包的每个元素映射到包中的索引 - 这是“索引序列技巧”的典型用例:

template <int... I> struct index_sequence {};
template <int N, int... I>
struct make_index_sequence : make_index_sequence<N-1,N-1,I...> {};
template <int... I>
struct make_index_sequence<0, I...> : index_sequence<I...> {};

template<typename... Values, int... I>
auto getColumns(index_sequence<I...>) ->
  decltype(std::make_tuple(getColumn<Values>(I)...)) {
    return std::make_tuple(getColumn<Values>(I)...);
}

template<typename... Values>
auto getColumns() ->
  decltype(getColumns<Values...>(make_index_sequence<sizeof...(Values)>())) {
    return getColumns<Values...>(make_index_sequence<sizeof...(Values)>());
}

Live demo at Coliru.

答案 1 :(得分:1)

也许使用auto

template<typename... Values>
auto getColumns() -> decltype(std::make_tuple(getColumns<Values>()...))
{
    return std::make_tuple(getColumns<Values>()...);
}

在C ++ 14中,您将能够省略-> decltype...部分,因为它将从函数体中推断出来。