为什么 C++20 范围适配器的 return 视图不是常量表达式?

Why the return view of C++20 range adaptors is not a constant expression?

考虑以下代码:

#include <ranges>

int main() {
  constexpr int a[] = {1, 2, 3, 4};
  constexpr auto r = a | std::views::take(3);
  static_assert(*r.begin() == 1);
}

msvc 接受它,gcc rejects 它与:

<source>:5:44: error: 'std::ranges::take_view<std::ranges::ref_view<const int [4]> >{3, std::ranges::ref_view<const int [4]>{(& a)}}' is not a constant expression
    5 |   constexpr auto r = a | std::views::take(3);
      |                                            ^

为什么r不是常量表达式?

范围位是一个红色鲱鱼。可以归结为这个

int main() {
  constexpr int a[] = {1, 2, 3, 4};
  constexpr auto r = a ;
}

我们无法形成指向 a 第一个元素的 constexpr 指针。这样的指针被限制为仅保存具有静态存储持续时间的对象的地址。视图对象(作为其在某个深处实现的一部分)将需要保留迭代器的副本(在我们的例子中是指针)。

由于 r 被声明为 constexpr 并且它在内部持有一个指向对象的指针,该对象也必须具有静态存储持续时间。的确,修改代码包含

static constexpr int a[] = {1, 2, 3, 4};

GCC accept成为你的榜样。

但是 MSVC 不符合标准。它也接受无效的普通指针示例。