我正在研究一个通用的(C ++ 11)代码,该代码应该与boost::multi_array
,Eigen::Matrix
以及可能的其他类型的n维数组一起使用。在几个点上,我需要访问给定数组类型的元素类型。 boost数组包含一个名为Element
的typedef,而Eigen数组包含一个名为Scalar
的typedef。
我想要的是一个返回给定数组类型的元素类型的类型特征。不幸的是,我不能只为所有可能的数组类型专门化trait类,因为Eigen使用表达式模板,因此,有无数种类型的特征矩阵。因此,我使用SFINAE和enable_if来实现我的特性。 enable_if应该选择的标准是,类型是否具有名为Element
的typedef,还是名为Scalar
的类型。
为此,我实现了类型特征has_element
和has_scalar
,它们确定是否存在相应的typedef。我的实现受到this blog post关于类型要求的启发。
template <class T>
using ToVoid = void;
template <class T, class Enable = void>
struct has_scalar : public std::false_type {};
template <class T>
struct has_scalar<T, ToVoid<typename T::Scalar>> : public std::true_type {};
template <class T, class Enable = void>
struct has_element : public std::false_type {};
template <class T>
struct has_element<T, ToVoid<typename T::Element>> : public std::true_type {};
这个想法是,如果没有提供typedef,编译器将选择false_type
版本,如果typedef存在,将选择更专业的 true_type
版本
获取标量类型的实际特征是这样实现的:
template <class Condition, class T = void>
using EnableIf = typename std::enable_if<Condition::value, T>::type;
template <class T, class Enable = void>
struct scalar_of;
template <class T>
struct scalar_of<T, EnableIf<has_element<T>>> {
using type = typename T::Element;
};
template <class T>
struct scalar_of<T, EnableIf<has_scalar<T>>> {
using type = typename T::Scalar;
};
template <class T>
using ScalarOf = typename scalar_of<T>::type;
一个简单的用例是:
struct BoostArray {
using Element = double;
};
struct EigenArray {
using Scalar = float;
};
int main() {
using std::is_same;
assert((is_same<double, ScalarOf<BoostArray>>::value));
assert((is_same<float, ScalarOf<EigenArray>>::value));
}
现在,奇怪的是,这对Clang 3.4来说效果很好。但是,GCC 4.8.1无法编译它并给出以下错误消息:
test.cpp: In substitution of ‘template<class T> using ScalarOf = typename scalar_of<T>::type [with T = BoostArray]’:
test.cpp:51:5: required from here
test.cpp:37:45: error: ambiguous class template instantiation for ‘struct scalar_of<BoostArray, void>’
using ScalarOf = typename scalar_of<T>::type;
^
test.cpp:27:8: error: candidates are: struct scalar_of<T, typename std::enable_if<has_element<T, void>::value, void>::type>
struct scalar_of<T, EnableIf<has_element<T>>> {
^
test.cpp:32:8: error: struct scalar_of<T, typename std::enable_if<has_scalar<T, void>::value, void>::type>
struct scalar_of<T, EnableIf<has_scalar<T>>> {
clang版本为here并且运行正常。 gcc版本为here,无法编译。
我的问题:我的代码是否正确,这是GCC 4.8.1的问题;或者我做错了什么,Clang在编译时太慷慨了?在任何情况下,我如何更改我的代码,以便GCC 4.8.1编译它?
答案 0 :(得分:10)
看起来是yet another manifestation的CWG issue 1558。该标准尚不清楚别名模板中未使用的模板参数是否会导致替换失败。 Clang将其视为替代失败; GCC只是忽略了未使用的参数。
使用
template<typename... Ts>
struct make_void { typedef void type;};
template<typename... Ts>
using ToVoid = typename make_void<Ts...>::type;