什么时候constexpr函数在编译时求值?
When does a constexpr function get evaluated at compile time?
既然可以在运行时调用声明为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函数,而编译器决定将其视为普通函数,而不是在编译期间工作?是否存在已知的常见陷阱?
constexpr
函数将在编译时被求值,当它的所有参数都是常量表达式并且结果也用于常量表达式时。常量表达式可以是文字(如42
),非类型模板参数(如template<class T, size_t N> class array;
中的N
), enum
元素声明(如enum Color { Red, Blue, Green };
中的Blue
),另一个声明的变量constexpr,等等。
They 可以在其所有参数都是常量表达式且结果不是用于常量表达式的时求值,但这取决于实现。
当需要常量表达式时,必须在编译时对函数求值。
保证这一点的最简单方法是使用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;
或:
std::cout << std::integral_constant<int, POW(i, 2)>::value << std::endl;
或
#define POW_C(base, power) (std::integral_constant<decltype(POW((base), (power)), POW((base), (power))>::value)
std::cout << POW_C(63, 2) << std::endl;
或
template<int base, int power>
struct POW_C {
static constexpr int value = POW(base, power);
};
std::cout << POW_C<2, 63>::value << std::endl;
相关文章:
- 如何确认我的constexpr表达式实际上已经在编译时执行
- 要与"if constexpr"一起使用的编译时消息(在预处理器之后)
- 用于循环编译的 constexpr
- 使用 constexpr 替换 #define 和 #ifdef 进行条件编译
- constexpr 使用 clang 编译 TensorFlow 时出错
- C++ constexpr语言 - 可以在编译时评估值吗?
- 编译时生成应在构造函数中创建的非 constexpr 对象数组
- constexpr 函数在编译时获取值,即使我的变量不是 constexpr
- 如何判断是否在编译时计算了"constexpr"(无需手动检查)
- C++编译时使用 constexpr 字符数组指针分配静态数组?
- Constexpr变量不是编译时值
- 在if constexpr中使用带参数包的概念时,升级到gcc 9后出现编译错误
- C++constexpr编译问题
- constexpr 函数中的 for 循环无法使用 MSVC 19.23 进行编译
- 为什么递归 constexpr 模板值无法编译?
- 从语言设计层面来看,当编译时无法推断条件时,为什么"if constexpr"不衰减到"trival if"
- C++17 在逗号上拆分 constexpr 字符串并在编译时有元素的数量?
- 这是通过初始化 constexpr 变量来标记编译错误的合理跳转
- 在编译时通过 constexpr 或模板函数获取多维 std::array 的大小
- MSVC 2015无法编译Constexpr Atan