【发布时间】:2019-10-27 09:50:01
【问题描述】:
所以我在这里写了一个答案:https://stackoverflow.com/a/56569397/2642059,它努力在编译时计算log2,如下所示:
template <unsigned int x>
constexpr enable_if_t<x != 0U, int> log2 = 1 + log2<x / 2U>;
template <>
constexpr int log2<1U> = 0;
这很好用,但我觉得我不应该专攻:
template <unsigned int x>
constexpr enable_if_t<x != 0U, int> log2 = x < 4U ? 1 : 1 + log2<x / 2U>;
但是这个gives me the error:
代替
template<bool _Cond, class _Tp> using enable_if_t = typename std::enable_if::type [with bool _Cond = (0u != 0u); _Tp = int]:
prog.cpp:7:61: 从constexpr std::enable_if_t<true, int> log2<4u>递归需要
prog.cpp:7:61: 需要来自constexpr std::enable_if_t<true, int> log2<8u>
prog.cpp:10:11:从这里需要 /usr/include/c++/6/type_traits:2523:61: 错误:struct std::enable_if<false, int>中没有名为type的类型
有什么方法可以防止编译器将递归展开得太远?
【问题讨论】:
-
有没有一种方法可以防止编译器将递归展开得太远? 是的,通过为不进行递归实例化的基本情况添加一个特化。
-
@MilesBudnek 我不希望为
log2<0>定义这个你是说有更好的方法吗? -
没关系,我把基本情况误读为 0。
enable_if是有道理的。 -
您的模板是递归的。对于我们生活中的任何递归,我们都需要一个叶子案例。在 TMP 中,提供叶案例的唯一方法是专门化。如果您将 TMP 替换为 constexpr 函数和 constexpr if,则可以避免专门化。
-
@JonathanMee 使用一些代码发布了相同效果的答案。
标签: c++ recursion metaprogramming template-specialization template-variables