Объедините несколько адаптеров диапазонов в один диапазон в C++20
Рассмотрим следующий случай:
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);
Это кажется невозможным, поскольку rng1
, rng2
, а также rng3
бывают очень разные типы.
Может кто подскажет альтернативное решение? Спасибо.
2 ответа
Да, то, что вы написали, только в другом пространстве имен - в стандартной библиотеке нет concat, но есть в диапазоне-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 .