【发布时间】:2012-12-24 06:26:46
【问题描述】:
既然声明为 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;
}
在这种情况下,i 在编译时是未知的,这可能是编译器将 POW() 视为在运行时调用的常规函数的原因。然而,这种动态,尽管看起来很方便,但有一些不切实际的含义。例如,是否存在我希望编译器在编译时计算 constexpr 函数的情况,而编译器决定将其视为普通函数,而它在编译时也可以工作?是否有任何已知的常见陷阱?
【问题讨论】:
-
AFAIK,当所有参数都是常量表达式时。
-
@chris 如果我写
POW((unsigned __int64)2, 63)会怎样。这仍然算作常量表达式吗? -
@chris:实际上,它比我想象的要复杂。我认为
constexpr仅在其结果用作模板参数、数组绑定或其他整数常量时才需要进行评估。任何其他时间都是优化。事实上,即使给定常量表达式参数,也可能需要在运行时执行。constexpr int func(int p) { return !p ? 1 : throw std::exception("HI");}必须在给定非零输入时在运行时进行评估。 -
作为常量表达式的初始化器构成静态初始化阶段的一部分,例如
constexpr int a = POW(5, 4);。这基本上是在编译时计算的。但是你当然仍然可以在其他地方使用POW。 -
@MooingDuck:除非你前面提到的常量表达式“requirerers”中使用了函数的结果,否则它会因为异常而产生编译时错误。
标签: c++ c++11 runtime compile-time constexpr