C++20常量表达式向量和字符串不起作用

2022-08-15 00:00:00 visual-c++ c++ c++20 stdstring constexpr

尝试创建constexprstd::stringstd::vector对象时遇到奇怪的编译器错误:

#include <vector>
#include <string>

int main()
{
    constexpr std::string cs{ "hello" };
    constexpr std::vector cv{ 1, 2, 3 };
    return 0;
}

编译器报告";表达式必须具有常量值:

我错过了什么吗?我使用的是最新的Microsoft Visual Studio 2019版本:16.11.4,参考(https://en.cppreference.com/w/cpp/compiler_support)说明此编译器版本支持constexpr字符串和向量:

我还尝试了constexpr std::array,它确实有效。该问题是否与向量关联的动态内存分配有关?


解决方案

您的程序实际上是格式错误的,尽管错误可能很难理解。constexprC++20中的分配支持是有限的-您只能进行临时分配。也就是说,必须在常量评估结束时完全释放分配。

所以您不能这样写:

int main() {
    constexpr std::vector<int> v = {1, 2, 3};
}

因为v的分配是持续的-它是非暂时性的。这就是错误告诉您的:

<source>(6): error C2131: expression did not evaluate to a constant
<source>(6): note: (sub-)object points to memory which was heap allocated during constant evaluation

v不能是常量,因为它仍在保留堆分配,而且不允许这样做。

但您可以写入this:

constexpr int f() {
    std::vector<int> v = {1, 2, 3};
    return v.size();
}

static_assert(f() == 3);

在这里,v的分配是暂时的-f()返回时将释放内存。但我们仍然可以在constexpr期间使用std::vector

相关文章