每个可变参数模板参数生成一个类成员

时间:2015-01-14 11:18:24

标签: c++ c++11 variadic-templates template-meta-programming

我有一个模板类,其中每个模板参数代表内部计算可以处理的一种类型的值。需要模板(而不是函数重载),因为值作为boost :: any传递,并且它们的类型在运行时之前不清楚。

为了正确地转换为正确的类型,我希望每个可变参数类型都有一个成员列表,如下所示:

template<typename ...AcceptedTypes> // e.g. MyClass<T1, T2>
class MyClass {
    std::vector<T1> m_argumentsOfType1;
    std::vector<T2> m_argumentsOfType2; // ...
};

或者,我想将模板参数类型存储在列表中,以便用它做一些RTTI魔术(?)。但是如何将它们保存在std :: initializer_list成员中对我来说也不清楚。

感谢您的帮助!

5 个答案:

答案 0 :(得分:17)

正如您已经暗示的那样,最好的方法是使用元组:

template<typename ...AcceptedTypes> // e.g. MyClass<T1, T2>
class MyClass {
    std::tuple<std::vector<AcceptedTypes>...> vectors;
};

这是将“字段”相乘的唯一方法,因为你无法神奇地拼写字段名称。另一个重要的事情可能是获得一些命名访问权限。我想你想要实现的是拥有多个具有唯一类型的向量,因此你可以通过其值类型“搜索”正确的向量:

template <class T1, class T2>
struct SameType
{
    static const bool value = false;
};

template<class T>
struct SameType<T, T>
{
    static const bool value = true;
};

template <typename... Types>
class MyClass
{
     public:
     typedef std::tuple<vector<Types>...> vtype;
     vtype vectors;

     template<int N, typename T>
     struct VectorOfType: SameType<T,
        typename std::tuple_element<N, vtype>::type::value_type>
     { };

     template <int N, class T, class Tuple,
              bool Match = false> // this =false is only for clarity
     struct MatchingField
     {
         static vector<T>& get(Tuple& tp)
         {
             // The "non-matching" version
             return MatchingField<N+1, T, Tuple,
                    VectorOfType<N+1, T>::value>::get(tp);
         }
     };

     template <int N, class T, class Tuple>
     struct MatchingField<N, T, Tuple, true>
     {
        static vector<T>& get(Tuple& tp)
        {
            return std::get<N>(tp);
        }
     };

     template <typename T>
     vector<T>& access()
     {
         return MatchingField<0, T, vtype,
                VectorOfType<0, T>::value>::get(vectors);
     }
};

以下是测试用例,您可以尝试一下:

int main( int argc, char** argv )
{
    int twelf = 12.5;
    typedef reference_wrapper<int> rint;

    MyClass<float, rint> mc;
    vector<rint>& i = mc.access<rint>();

    i.push_back(twelf);

    mc.access<float>().push_back(10.5);

    cout << "Test:\n";
    cout << "floats: " << mc.access<float>()[0] << endl;
    cout << "ints: " << mc.access<rint>()[0] << endl;
    //mc.access<double>();

    return 0;
}

如果您使用的任何类型都不是您传递给myClass的类型列表(请参阅此注释的双重访问权限),您将收到编译错误,不太可读,但gcc至少指出导致问题的正确位置,至少有这样的错误消息表明问题的正确原因 - 例如,如果您尝试执行 mc.access&lt; double&gt;()

 error: ‘value’ is not a member of ‘MyClass<float, int>::VectorOfType<2, double>’

答案 1 :(得分:5)

在πάντα-ῥεῖ的评论中提到的一种做这种事情的方法是使用元组。他没有解释(可能是为了拯救你自己)是看起来的样子。

以下是一个例子:

using namespace std;

// define the abomination    
template<typename...Types>
struct thing
{
    thing(std::vector<Types>... args)
    : _x { std::move(args)... }
    {}

    void print()
    {
        do_print_vectors(std::index_sequence_for<Types...>());
    }

private:
    template<std::size_t... Is>
    void do_print_vectors(std::index_sequence<Is...>)
    {
        using swallow = int[];
        (void)swallow{0, (print_one(std::get<Is>(_x)), 0)...};
    }

    template<class Vector>
    void print_one(const Vector& v)
    {
        copy(begin(v), end(v), ostream_iterator<typename Vector::value_type>(cout, ","));
        cout << endl;
    }

private:
    tuple<std::vector<Types>...> _x;
};


// test it
BOOST_AUTO_TEST_CASE(play_tuples)
{
    thing<int, double, string> t {
        { 1, 2, 3, },
        { 1.1, 2.2, 3.3 },
        { "one"s, "two"s, "three"s }
    };

    t.print();
}

预期产出:

1,2,3,
1.1,2.2,3.3,
one,two,three,

答案 2 :(得分:4)

一种不使用元组的替代解决方案是使用CRTP创建类层次结构,其中每个基类都是其中一种类型的特化:

#include <iostream>
#include <string>

template<class L, class... R> class My_class;

template<class L>
class My_class<L>
{
public:

protected:
  L get()
  {
    return val;
  }

  void set(const L new_val)
  {
    val = new_val;
  }

private:
  L val;
};

template<class L, class... R>
class My_class : public My_class<L>, public My_class<R...>
{
public:
  template<class T>
  T Get()
  {
    return this->My_class<T>::get();
  }

  template<class T>
  void Set(const T new_val)
  {
    this->My_class<T>::set(new_val);
  }
};

int main(int, char**)
{
  My_class<int, double, std::string> c;
  c.Set<int>(4);
  c.Set<double>(12.5);
  c.Set<std::string>("Hello World");

  std::cout << "int: " << c.Get<int>() << "\n";
  std::cout << "double: " << c.Get<double>() << "\n";
  std::cout << "string: " << c.Get<std::string>() << std::endl;

  return 0;
}

答案 3 :(得分:1)

有人建议使用这种直观的语法P1858R1 Generalized pack declaration and usage进行这种扩展。您还可以初始化成员并通过索引访问它们。您甚至可以通过编写using... tuple_element = /*...*/来支持结构化绑定:

template <typename... Ts>
class MyClass {
    std::vector<Ts>... elems;
public:
    using... tuple_element = std::vector<Ts>;

    MyClass() = default;
    explicit MyClass(std::vector<Ts>... args) noexcept
        : elems(std::move(args))...
    {
    }

    template <std::size_t I>
        requires I < sizeof...(Ts)
    auto& get() noexcept
    {
        return elems...[I];
    }

    template <std::size_t I>
        requires I < sizeof...(Ts)
    const auto& get() const
    {
        return elems...[I];
    }

    // ...
};

然后可以像这样使用该类:

using Vecs = MyClass<int, double>;

Vecs vecs{};
vecs.[0].resize(3, 42);

std::array<double, 4> arr{1.0, 2.0, 4.0, 8.0};
vecs.[1] = {arr.[:]};

// print the elements
// note the use of vecs.[:] and Vecs::[:]
(std::copy(vecs.[:].begin(), vecs.[:].end(),
           std::ostream_iterator<Vecs::[:]>{std::cout, ' '},
 std::cout << '\n'), ...);

答案 4 :(得分:0)

使用boost::variant

,这是一个不太完美有效的实现
template<typename ... Ts>
using variant_vector = boost::variant< std::vector<Ts>... >;

template<typename ...Ts>
struct MyClass {
  using var_vec = variant_vector<Ts...>;
  std::array<var_vec, sizeof...(Ts)> vecs;
};

我们创建了一个variant-vector,它可以包含其中一个类型列表。您必须使用boost::variant来获取内容(这意味着了解内容的类型或撰写访问者)。

然后我们存储这些变体向量的数组,每种类型一个。

现在,如果您的类只保留一种类型的数据,则可以取消该数组,并且只有一个var_vec类型的成员。

我不明白为什么你想要每种类型的一个向量。我可以看到想要一个向量,其中每个元素都是任何类型之一。这将是vector<variant<Ts...>>,而不是上面的variant<vector<Ts>...>

variant<Ts...>是与boost联合的类型。 any聪明的boost - void*optionalboost那里或没有。

template<class...Ts>
boost::optional<boost::variant<Ts...>> to_variant( boost::any );

可能是一个有用的函数,它接受any并尝试将其转换为Ts...中的任何variant类型,如果成功则返回它(并返回如果没有,则为空optional