在 C++20 中将多个范围适配器连接到一个范围中

Concatenate multiple range adaptors into one single ranges in C++20

本文关键字:范围 一个 连接 C++20 适配器      更新时间:2023-10-16

考虑以下情况:

std::vector<int> v{0, 1, 2, 3, 4, 5};
// 0 1 2 3 4 5
auto rng1 = std::views::all(v);
// 5 4 3 2 1 0
auto rng2 = std::views::reverse(v);
// 4 2 0
auto rng3 = std::views::filter(rng2, [](int x){return x % 2 == 0;});

有没有一种优雅的方法将这三个适配器连接到一个视图中,如下所示:

// 0 1 2 3 4 5 5 4 3 2 1 0 4 2 0
auto final_rng = std::views::concat(rng1, rng2, rng3);

这似乎是不可能的,因为rng1rng2rng3是非常不同的类型。

有人可以给出替代解决方案吗?谢谢。

是的,您只是在不同的命名空间中编写的内容 - 标准库中没有 concat,但在 range-v3 中有一个:

auto final_rng = ranges::views::concat(rng1, rng2, rng3);

范围是不同类型的事实并不构成问题。你只有一个迭代器,它由范围的基础迭代器的变体构建而成。重要的部分是范围的值类型是相同的 - 在这里,所以这完全没问题。

标准中实际上有一个 concat(( 与 C++23 2/3 一起提供。查看规格: https://www.open-std.org/jtc1/sc22/wg21/docs/papers/2020/p2214r0.html