什么时候constexpr函数在编译时得到评估?

Byz*_*ian 58 c++ runtime compile-time constexpr c++11

由于在运行时可能会调用声明为constexpr的函数,编译器在哪个条件下决定是在编译时还是在运行时计算它?

template<typename base_t, typename expo_t>
constexpr base_t POW(base_t base, expo_t expo)
{
    return (expo != 0 )? base * POW(base, expo -1) : 1;
}

int main(int argc, char** argv)
{
    int i = 0;
    std::cin >> i;

    std::cout << POW(i, 2) << std::endl;
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

在这种情况下,我在编译时是未知的,这可能是编译器将POW()视为在运行时调用的常规函数​​的原因.然而,这种动态虽然看起来很方便,但却具有一些不切实际的含义.例如,是否有一种情况我希望编译器在编译时计算constexpr函数,编译器决定将其视为普通函数,而它在编译时也会工作?有任何已知的常见陷阱吗?

K-b*_*llo 85

constexpr函数在编译时进行评估,当它的所有参数都是常量表达式时,结果也用在常量表达式中.常量表达式可以是文字(如42),非类型模板参数(如Nin template<class T, size_t N> class array;),enum元素声明(如Bluein enum Color { Red, Blue, Green };,另一个变量声明为constexpr,依此类推).

当它们的所有参数都是常量表达式并且结果不在常量表达式中使用时,可能会对它们进行求值,但这取决于实现.

  • @mcmcc:对于你的"别的东西",为什么不编写一个接受浮点值(或实际上你喜欢的任何类型)的`constexpr`函数并返回一个整数,然后在模板非类型参数中使用它.`constexpr size_t check_nonintegral_constexpr(float v){return sizeof v; } std :: array <check_nonintegral_constexpr(pow(2.0,4))>断言;` (6认同)
  • 如果所有参数都是常量表达式,那么在标准中是否保证在编译时评估`constexpr`函数?我相信标准所说的唯一的事情是`constexpr`函数可以在必须在编译时评估的上下文中使用,例如模板参数.还有其他任何事情由编译器来决定.至少这是我迄今为止所相信的. (4认同)
  • @mcmcc尝试将结果用作非类型模板参数。这应该涵盖完整的constexpr。对于浮球,您还需要其他东西。 (2认同)

Pub*_*bby 19

当需要常量表达式时,必须在编译时评估该函数.

保证这一点的最简单方法是使用constexpr值,或std::integral_constant:

constexpr auto result = POW(i, 2); // this should not compile since i is not a constant expression
std::cout << result << std::endl;
Run Code Online (Sandbox Code Playgroud)

要么:

std::cout << std::integral_constant<int, POW(i, 2)>::value << std::endl;
Run Code Online (Sandbox Code Playgroud)

要么

#define POW_C(base, power) (std::integral_constant<decltype(POW((base), (power)), POW((base), (power))>::value)

std::cout << POW_C(63, 2) << std::endl;
Run Code Online (Sandbox Code Playgroud)

要么

template<int base, int power>
struct POW_C {
  static constexpr int value = POW(base, power);
};

std::cout << POW_C<2, 63>::value << std::endl;
Run Code Online (Sandbox Code Playgroud)

  • @balki这是我想出来的,但我不确定它是否是最好的解决方案.我也不确定是否返回rvalue引用实际上是这样的.`#define FORCE_CT_EVAL(func)[](){constexpr auto ___expr = func; return std :: move(___ expr);}()` (5认同)