C++20 介绍 views::all
这是一个范围适配器,返回 view
这包括其范围参数的所有元素。
表达式 views::all(E)
是表达式等效(具有相同的效果)到:
decay-copy(E)
如果E
的衰变类型型号 view
. ref_view{E}
如果该表达式格式正确 subrange{E}
第一种情况代表一个
view
的类型在使用 views::all
管道传输后不会改变( goldbot ):auto r = views::iota(0);
static_assert(std::same_as<decltype(r), decltype(r | views::all)>);
第二种情况用于包装一个 viewable_range
与 ref_view
为方便范围管道操作:int r[] = {0, 1, 2};
static_assert(std::same_as<ranges::ref_view<int[3]>, decltype(r | views::all)>);
但是关于第三种情况,我想不出是在什么情况下subrange{E}
格式良好且ref_view{E}
是畸形的。它的目的是什么?有人可以举一个例子吗?
最佳答案
But regarding the third case, I can't think of under what circumstances
subrange{E}
is well-formed andref_view{E}
is ill-formed.
ref_view{E}
仅适用于左值范围。subrange{E}
仅适用于借用范围。您可以在 [range.subrange.general] 中找到其扣除指南。 :template<borrowed_range R>
subrange(R&&) ->
subrange<iterator_t<R>, sentinel_t<R>,
(sized_range<R> || sized_sentinel_for<sentinel_t<R>, iterator_t<R>>)
? subrange_kind::sized : subrange_kind::unsized>;
借用范围再次是左值或选择从中借用的范围。类型如 string_view
和 span
例如,是借来的。所以如果你有类似右值的东西
vector<int>
,那么这不是 View (第一项),也不能构建 ref_view
从它(因为它是一个右值),你也不能构造一个 subrange
从它(因为它是一个非借用的范围)。我意识到这并不能完全回答问题,因为我在输入答案时颠倒了头脑中的极性。但是T.C.'s got me covered .
另一个纯粹假设的例子是一个非 View 范围,它将其内容存储在
shared_ptr
中。 ,然后它的迭代器也共享该数据。就像是:struct SharedVector {
std::shared_ptr<std::vector<int>> data;
struct Iterator {
std::shared_ptr<std::vector<int>> data;
std::vector<int>::iterator cur;
// ...
};
auto begin() -> Iterator { return {data, data->begin()}; }
auto end() -> Iterator { return {data, data->end()}; }
};
右值 SharedVector
不会是一个 View (不是 O(1)
-可破坏的),你不能 ref_view{E}
它(因为它是一个右值),但这样的范围仍然可以借用,所以 subrange{E}
可以工作。
关于c++ - 在什么情况下 ref_view{E} 格式错误而 subrange{E} 不是?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67466714/