为什么 Clang 12 拒绝以 C++20 的方式初始化聚合?

Why does Clang 12 refuse to initialize aggregates in the C++20 way?

据我了解,以下程序应该可以在 C++20 模式下运行:

#include <vector>

struct B{ int a0, a1; };

int main()
{
    std::vector<B> bs;
    bs.emplace_back( 0, 0 );
}

在 Visual Studio 2019 和 gcc 11 中确实如此。但在 clang 12 中却没有,它会产生错误:

/opt/compiler-explorer/gcc-snapshot/lib/gcc/x86_64-linux-gnu/12.0.0/../../../../include/c++/12.0.0/bits/alloc_traits.h:514:4: error: no matching function for call to 'construct_at'
          std::construct_at(__p, std::forward<_Args>(__args)...);
          ^~~~~~~~~~~~~~~~~

在在线编译器中:https://gcc.godbolt.org/z/GzccTWc5z

这是因为clang还没有完全支持C++20吗?

这是一项 C++20 功能,允许通过标准构造函数语法而不是典型的花括号列表初始化语法进行聚合初始化。 (请注意,这仅在无法在对默认或 copy/move 构造函数的有效调用中使用参数时才有效。如果可以,将调用它而不是执行聚合初始化。)

根据官方C++ Support in Clang page, Clang does not yet support parenthesized initialization of aggregates (which is P0960R3 and P1975R0)。这是从 Clang 版本 13 开始的。

还维护了 C++20 功能的支持矩阵 on cppreference.com。由此可见对P0960R3的支持如下:

  • GCC:从版本 10 开始支持
  • MSVC:从版本 19.28 开始受支持
  • EDG eccp:从 5.1 版开始支持
  • Clang 和 Apple Clang:不支持
  • 英特尔 ICC:不支持