这实际上是可行的,使用c++11特性。
是的,initializer_list希望它的所有元素都是相同类型的。技巧在于我们可以创建一个包装类,可以static_cast
成我们想要的所有类型。这很容易实现:
template <typename... tlist>
class MultiTypeWrapper {
};
template <typename H>
class MultiTypeWrapper<H> {
public:
MultiTypeWrapper() {}
MultiTypeWrapper(const H &value) : value_(value) {}
operator H () const {
return value_;
}
private:
H value_;
};
template <typename H, typename... T>
class MultiTypeWrapper<H, T...>
: public MultiTypeWrapper<T...> {
public:
MultiTypeWrapper() {}
MultiTypeWrapper(const H &value) : value_(value) {}
template <typename C>
MultiTypeWrapper(const C &value) : MultiTypeWrapper<T...>(value) {}
operator H () const {
return value_;
}
private:
H value_;
};
使用隐式转换构造函数,我们可以将类MultiTypeWrapper的初始化列表(或隐式转换为初始化列表的向量)传递给像{1,2.5,'c',4}这样的内容。这意味着我们不能编写一个接受此类初始化列表作为参数的函数,如下所示:
template <typename... T>
std::tuple<T...> create_tuple(std::vector<unit_test::MultiTypeWrapper<T...> > init) {
....
}
我们使用另一种技巧将向量中的每个值转换为其原始类型(请注意,在MultiTypeWrapper
的定义中,我们提供了隐式转换),并将其分配给元组中的相应插槽。这就像模板参数上的递归:
template <int ind, typename... T>
class helper {
public:
static void set_tuple(std::tuple<T...> &t, const std::vector<MultiTypeWrapper<T...> >& v) {
std::get<ind>(t) = static_cast<typename std::tuple_element<ind,std::tuple<T...> >::type>(v[ind]);
helper<(ind-1),T...>::set_tuple(t,v);
}
};
template <typename... T>
class helper<0, T...> {
public:
static void set_tuple(std::tuple<T...> &t, const std::vector<MultiTypeWrapper<T...> >& v) {
std::get<0>(t) = static_cast<typename std::tuple_element<0,std::tuple<T...> >::type>(v[0]);
}
};
template <typename... T>
std::tuple<T...> create_tuple(std::vector<unit_test::MultiTypeWrapper<T...> > init) {
std::tuple<T...> res;
helper<sizeof...(T)-1, T...>::set_tuple(res, init);
return res;
}
请注意,我们需要创建
set_tuple
的辅助类,因为c++不支持函数特化。现在,如果我们想测试代码:
auto t = create_tuple<int,double,std::string>({1,2.5,std::string("ABC")});
printf("%d %.2lf %s\n", std::get<0>(t), std::get<1>(t), std::get<2>(t).c_str());
输出结果将会是:
1 2.50 ABC
这是在我的桌面上使用clang 3.2进行测试的。
希望我的输入能有所帮助 :)