首页 > 解决方案 > 使用 std::enable_if 的多变量模板特化

问题描述

我试图用这些有效值简洁地定义一个变量模板:

// (template<typename T> constexpr T EXP = std::numeric_limits<T>::max_exponent / 2;)

// float and double scalar definitions:
const double huge = std::scalbn(1, EXP<double>);
const float huge = std::scalbn(1, EXP<float>);
// SIMD vector definitions:
const Vec8f huge = Vec8f(huge<float>); // vector of 8 floats
const Vec8d huge = Vec8d(huge<double>); // vector of 8 doubles
const Vec4f huge = Vec4f(huge<float>); // vector of 4 floats
// Integral types should fail to compile

VecXX 向量定义(SIMD 向量)需要使用如图所示的相应标量类型(例如huge<float>,对于 s 的向量float)。这可以作为VecXX::value_type或通过类型特征样式模板类 ( VectorTraits<VecXX>::value_type) 获得。

理想情况下,我想我会有类似的东西:

// Primary. What should go here? I want all other types to not compile
template<typename T, typename Enabler = void>
const T huge = T{ 0 };

// Scalar specialization for floating point types:
template<typename T>
const T huge<T> = std::enable_if_t<std::is_floating_point<T>::value, T>(std::scalbn(1, EXP<T>));

// Vector specialization, uses above declaration for corresponding FP type
template<typename T>
const T huge<T> = std::enable_if_t<VectorTraits<T>::is_vector, T>(huge<VectorTraits<T>::scalar_type>);

但我不能完全弄清楚一个工作版本(上面失败了“重新定义const T huge<T>”)。最好的方法是什么?

标签: c++c++14sfinaeenable-ifvariable-templates

解决方案


不完全是您所问的,但我希望以下示例可以向您展示如何使用 SFINAE 来专门化模板变量

template <typename T, typename = void>
constexpr T huge = T{0};

template <typename T>
constexpr T huge<T, std::enable_if_t<std::is_floating_point<T>{}>> = T{1};

template <typename T>
constexpr T huge<std::vector<T>> = T{2};

你可以检查它

std::cout << huge<int> << std::endl;
std::cout << huge<long> << std::endl;
std::cout << huge<float> << std::endl;
std::cout << huge<double> << std::endl;
std::cout << huge<long double> << std::endl;
std::cout << huge<std::vector<int>> << std::endl;

推荐阅读