当前位置: 首页 > 知识库问答 >
问题:

在编译时,何时计算Constexpr函数?

夏侯野
2023-03-14

由于声明为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()视为在运行时调用的常规函数的原因。然而,这种动态,尽管看起来很方便,却有一些不切实际的含义。例如,会不会有这样一种情况,我希望编译器在编译时计算一个常数表达式函数,而编译器决定将其视为正常函数,而它在编译时也可以工作?有什么已知的常见陷阱吗?

共有2个答案

顾英发
2023-03-14

当需要常量表达式时,必须在编译时对函数求值。

保证这一点的最简单的方法是使用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;
沈宇定
2023-03-14

当所有参数都是常量表达式并且结果也在常量表达式中使用时,将在编译时计算函数。常量表达式可以是文字(如42),非类型模板参数(如模板中的N

当其所有参数都是常量表达式且结果未在常量表达式中使用时,可以对其求值,但这取决于实现。

 类似资料:
  • 我使用的是MSVS c 17,下面的代码无法编译: 在函数“makeIndices”中,我标记了出现错误的地方。它看起来像: 错误C2131:表达式未计算为常量 注意:失败是由于在变量的生命周期之外读取了变量导致的 注:参见“pred”的用法 注意:请参阅对函数模板实例化“auto makeIndices”的引用 上面的代码在GCC(Link)中编译并正常工作。 但是如何为MSVS修复它呢?

  • 我有以下几点: 操作是在编译时还是在运行时完成的?换句话说,在运行时,上述代码段和以下代码段之间是否存在性能差异: 编辑:我的问题不同于Java编译器是否会预先计算文字的总和?,因为我在算术运算中混合使用变量和文字。虽然差别很小,但正如@TagirValeev在评论中指出的(对文本的算术运算是在编译时还是在运行时计算的?),有些情况下,某些文字没有预先编译,即使它们可能是。

  • 我试着实现了一个列表容器,并决定将一些通用函数如< code>sum()移到基类,这样我就可以在其他容器中重用它们。 所有的基本支持类需要的是三个方法 empty()、 和 。我不能使这些纯粹的虚拟,因为支持类永远不会被实例化。但它仍然必须使用这些方法来实现自己的方法,如 我试过这样的东西: 但是尝试使用< code>sum()会导致编译错误 对于、和中的每一个。 有什么建议吗?

  • 下面是代码 在类中,我使用参数10/3调用方法(如代码中所示)。由于callMtd在baseClass中重载,默认情况下应该调用哪个版本?因为它是重载的,方法绑定应该在编译时发生,但是10/3的计算能在编译时发生吗?

  • 问题内容: 在最近的一次采访中,有人问我一个非常奇怪的问题。面试官问我如何仅使用编译器功能来计算1 + 2 + 3 + … + 1000。这意味着我不允许编写程序并执行它,但我只应该编写一个程序,该程序可以驱动编译器在编译时计算此和,并在编译完成时打印结果。作为提示,他告诉我,我可能会使用编译器的泛型和预处理器功能。可以使用C ++,C#或Java编译器。有任何想法吗??? 此问题与此处未询问任何

  • 为什么 C 编译器可以将函数声明为 constexpr,而 constexpr 不能是 constexpr? 例如:http://melpon.org/wandbox/permlink/AGwniRNRbfmXfj8r 输出: 为什么此行出错: