| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236 |
- // Range v3 library
- //
- // Copyright Eric Niebler 2014-present
- //
- // Use, modification and distribution is subject to the
- // Boost Software License, Version 1.0. (See accompanying
- // file LICENSE_1_0.txt or copy at
- // http://www.boost.org/LICENSE_1_0.txt)
- //
- // Project home: https://github.com/ericniebler/range-v3
- #include <iostream>
- #include <forward_list>
- #include <list>
- #include <vector>
- #include <range/v3/range/conversion.hpp>
- #include <range/v3/view/chunk.hpp>
- #include <range/v3/view/cycle.hpp>
- #include <range/v3/view/filter.hpp>
- #include <range/v3/view/iota.hpp>
- #include <range/v3/view/join.hpp>
- #include <range/v3/view/move.hpp>
- #include <range/v3/view/repeat.hpp>
- #include <range/v3/view/reverse.hpp>
- #include "../simple_test.hpp"
- #include "../test_iterators.hpp"
- #include "../test_utils.hpp"
- using namespace ranges;
- namespace
- {
- void test_input_ranges()
- {
- int const ints[] = {0,1,2,3,4};
- static constexpr auto N = ranges::size(ints);
- static constexpr auto K = 2;
- auto make_range = [&]{ return debug_input_view<int const>{ints} | views::chunk(K); };
- auto rng = make_range();
- using Rng = decltype(rng);
- CPP_assert(input_range<Rng>);
- CPP_assert(view_<Rng>);
- CPP_assert(!forward_range<Rng>);
- CPP_assert(sized_range<Rng>);
- CHECK(ranges::size(rng) == (N + K - 1) / K);
- CPP_assert(sized_sentinel_for<sentinel_t<Rng>, iterator_t<Rng>>);
- CHECK((ranges::end(rng) - ranges::begin(rng)) == int((N + K - 1) / K));
- rng = make_range();
- auto i = ranges::begin(rng);
- auto e = ranges::end(rng);
- CHECK(i != e);
- if (i == e) return;
- {
- auto r = *i;
- CHECK(ranges::size(r) == 2u);
- auto ii = ranges::begin(r);
- auto ee = ranges::end(r);
- CHECK(ii != ee);
- if (ii == ee) return;
- CHECK((ee - ii) == 2);
- CHECK(*ii == 0);
- CHECK(++ii != ee);
- if (ii == ee) return;
- CHECK((ee - ii) == 1);
- CHECK(*ii == 1);
- CHECK(++ii == ee);
- CHECK((ee - ii) == 0);
- }
- CHECK(++i != e);
- if (i == e) return;
- ::check_equal(*i, {2,3});
- CHECK(++i != e);
- if (i != e)
- {
- ::check_equal(*i, {4});
- CHECK(++i == e);
- }
- }
- }
- int main()
- {
- {
- auto v = views::iota(0,11) | to<std::vector>();
- auto rng1 = v | views::chunk(3);
- CPP_assert(random_access_range<decltype(rng1)>);
- CPP_assert(sized_range<decltype(rng1)>);
- auto it1 = ranges::begin(rng1);
- ::check_equal(*it1++, {0,1,2});
- ::check_equal(*it1++, {3,4,5});
- ::check_equal(*it1++, {6,7,8});
- ::check_equal(*it1++, {9,10});
- CHECK(it1 == ranges::end(rng1));
- ::check_equal(*ranges::next(it1, -3), {3,4,5});
- CHECK(size(rng1), 4u);
- CHECK(sizeof(rng1.begin()) == sizeof(v.begin()) * 2 + sizeof(std::ptrdiff_t) * 2);
- }
- {
- auto l = views::iota(0,11) | to<std::forward_list>();
- auto rng2 = l | views::chunk(3);
- CPP_assert(forward_range<decltype(rng2)>);
- CPP_assert(!bidirectional_range<decltype(rng2)>);
- CPP_assert(!sized_range<decltype(rng2)>);
- auto it2 = ranges::begin(rng2);
- ::check_equal(*it2++, {0,1,2});
- ::check_equal(*it2++, {3,4,5});
- ::check_equal(*it2++, {6,7,8});
- ::check_equal(*it2++, {9,10});
- CHECK(it2 == ranges::end(rng2));
- CHECK(sizeof(rng2.begin()) == sizeof(l.begin()) * 2 + sizeof(std::ptrdiff_t));
- }
- {
- // An infinite, cyclic range with cycle length == 1
- auto fives = views::repeat(5);
- CPP_assert(random_access_range<decltype(fives)>);
- auto rng = fives | views::chunk(3);
- CPP_assert(random_access_range<decltype(rng)>);
- auto it = rng.begin();
- auto it2 = next(it,3);
- CHECK((it2 - it) == 3);
- ::check_equal(*it, {5,5,5});
- ::check_equal(*it2, {5,5,5});
- }
- {
- // An infinite, cyclic range with cycle length == 3
- int const ints[] = {0,1,2};
- auto cyc = ints | views::cycle;
- //[0,1],[2,0],[1,2],[0,1],[2,0],[1,2],
- auto rng = cyc | views::chunk(2);
- CPP_assert(random_access_range<decltype(rng)>);
- auto it = rng.begin();
- auto it2 = next(it,2);
- ::check_equal(*it, {0,1});
- ::check_equal(*it2, {1,2});
- CHECK((it - it) == 0);
- CHECK((next(it,1) - it) == 1);
- CHECK((next(it,2) - it) == 2);
- CHECK((next(it,3) - it) == 3);
- CHECK((next(it,4) - it) == 4);
- CHECK((next(it,5) - it) == 5);
- CHECK((next(it,6) - it) == 6);
- CHECK((next(it,7) - it) == 7 );
- }
- {
- // An infinite, cyclic range with cycle length == 3
- int const ints[] = {0,1,2};
- auto cyc = ints | views::cycle;
- //[0,1,2,0],[1,2,0,1],[2,0,1,2],...
- auto rng = cyc | views::chunk(4);
- CPP_assert(random_access_range<decltype(rng)>);
- auto it = rng.begin();
- auto it2 = next(it,2);
- ::check_equal(*it, {0,1,2,0});
- ::check_equal(*it2, {2,0,1,2});
- // Strange, but not wrong necessarily:
- CHECK((it - it) == 0);
- CHECK((next(it,1) - it) == 1);
- CHECK((next(it,2) - it) == 2);
- CHECK((next(it,3) - it) == 3);
- CHECK((next(it,4) - it) == 4);
- CHECK((next(it,5) - it) == 5);
- CHECK((next(it,6) - it) == 6);
- CHECK((next(it,7) - it) == 7);
- }
- {
- // An infinite, cyclic range with cycle length == 10
- int const ints[] = {0,1,2,3,4,5,6,7,8,9};
- auto cyc = ints | views::cycle;
- auto rng = cyc | views::chunk(3);
- CPP_assert(random_access_range<decltype(rng)>);
- //[0,1,2],[3,4,5],[6,7,8],[9,0,1],[2,3,4],...
- auto it = rng.begin();
- auto it2 = next(it,2);
- ::check_equal(*it, {0,1,2});
- ::check_equal(*it2, {6,7,8});
- // Strange, but not wrong necessarily:
- CHECK((it - it) == 0);
- CHECK((next(it,1) - it) == 1);
- CHECK((next(it,2) - it) == 2);
- CHECK((next(it,3) - it) == 3);
- CHECK((next(it,4) - it) == 4);
- CHECK((next(it,5) - it) == 5);
- CHECK((next(it,6) - it) == 6);
- CHECK((next(it,7) - it) == 7);
- CHECK((next(it,8) - it) == 8);
- CHECK((next(it,9) - it) == 9);
- CHECK((next(it,10) - it) == 10);
- CHECK((next(it,11) - it) == 11);
- CHECK((next(it,12) - it) == 12);
- CHECK((next(it,13) - it) == 13);
- }
- test_input_ranges();
- {
- // Regression test for #567
- std::vector<std::vector<int>> vec{{1, 2, 3}, {4, 5, 6}};
- auto rng = vec | views::join | views::chunk(2);
- CPP_assert(input_range<decltype(rng)>);
- CPP_assert(input_range<range_reference_t<decltype(rng)>>);
- int const expected[][2] = {{1, 2}, {3, 4}, {5, 6}};
- ::check_equal(rng, expected);
- }
- {
- // Regression test for not-exactly #567 (https://github.com/ericniebler/range-v3/issues/567#issuecomment-315148392)
- int some_ints[] = {0,1,2,3};
- int const expected[][2] = {{0, 1}, {2, 3}};
- auto rng = views::all(some_ints);
- ::check_equal(rng | views::chunk(2), expected);
- }
- {
- // Regression test for https://stackoverflow.com/questions/49210190
- auto rng = views::closed_iota(1,25)
- | views::filter([](int item){ return item % 10 != 0; })
- | views::chunk(10);
- auto it = ranges::begin(rng);
- auto last = ranges::end(rng);
- CHECK(it != last);
- ::check_equal(*it, {1,2,3,4,5,6,7,8,9,11});
- CHECK(++it != last);
- ::check_equal(*it, {12,13,14,15,16,17,18,19,21,22});
- CHECK(++it != last);
- ::check_equal(*it, {23,24,25});
- CHECK(++it == last);
- }
- return ::test_result();
- }
|