iterator: improve reference semantics

This commit is contained in:
Danny Robson 2018-03-27 15:49:47 +11:00
parent 93185775e6
commit 1b023f7c8d
3 changed files with 127 additions and 103 deletions

View File

@ -22,6 +22,7 @@
#include "variadic.hpp" #include "variadic.hpp"
#include "view.hpp" #include "view.hpp"
#include <iterator>
template <typename Base> template <typename Base>
class referencing_iterator { class referencing_iterator {
@ -283,53 +284,22 @@ namespace util {
/////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////
namespace detail::zip { namespace detail::zip {
// holds a tuple of iterators for begin and end, and returns an
// iterator that transforms these iterators into tuples of value_types.
//
// this must be expressed in terms of iterators, rather than containers,
// because it dramatically simplifies iterating over raw arrays.
//
// IteratorT: a tuple of iterators across all containers
//
// StoreT: a tuple of containers we own. used when we were provided
// with an rval at zip time. allows us to destroy the data
// when we're actually done iterating.
template < template <
typename IteratorT, typename IteratorT,
typename StoreT, typename = std::make_index_sequence<std::tuple_size_v<IteratorT>>
typename I = std::make_index_sequence<std::tuple_size<IteratorT>::value>
> >
class collection; struct iterator;
//--------------------------------------------------------------------- template <typename IteratorT, std::size_t ...Indices>
template < struct iterator<IteratorT, std::index_sequence<Indices...>> {
typename IteratorT,
typename StoreT,
std::size_t ...I
>
class collection<
IteratorT,
StoreT,
std::index_sequence<I...>
> {
public: public:
collection (IteratorT _begin, IteratorT _end, StoreT &&_store): // we cannot be a forward iterator because we don't want to supply
m_begin { _begin }, // references to a value type as that would necessitate storing
m_end { _end }, // said value types within the iterator.
m_store { std::forward<StoreT> (_store) } using iterator_category = std::input_iterator_tag;
{ ; } using difference_type = std::ptrdiff_t;
struct iterator : std::iterator<
std::forward_iterator_tag,
std::tuple<
typename std::iterator_traits<
typename std::tuple_element<I,IteratorT>::type
>::value_type...
>,
std::size_t
> {
public:
iterator (IteratorT _iterators): iterator (IteratorT _iterators):
m_iterators (_iterators) m_iterators (_iterators)
{ ; } { ; }
@ -338,21 +308,17 @@ namespace util {
iterator& iterator&
operator++ (void) operator++ (void)
{ {
// HACK: we don't actually need to create a tuple here, std::tuple (++std::get<Indices> (m_iterators)...);
// but it's a zero cost method to expand the parameter
// pack.
std::make_tuple (++std::get<I> (m_iterators)...);
return *this; return *this;
} }
iterator operator++ (int); iterator operator++ (int);
auto auto
operator* (void) operator* (void)
{ {
return std::make_tuple (*std::get<I> (m_iterators)...); return std::forward_as_tuple (*std::get<Indices> (m_iterators)...);
} }
@ -374,24 +340,60 @@ namespace util {
}; };
iterator // holds a tuple of iterators for begin and end, and returns an
begin (void) // iterator that transforms these iterators into tuples of value_types.
{ //
return iterator { { std::get<I> (m_begin)... } }; // this must be expressed in terms of iterators, rather than containers,
} // because it dramatically simplifies iterating over raw arrays.
//
// we have to store begin and end iterators because we might not have
// enough information to determine the correct types from StoreT;
// eg, in the case of arrays that have decayed to pointers we can't
// find the end.
//
// BeginT: a tuple of begin iterators across all containers
//
// EndT: a tuple of end iterators across all containers
//
// StoreT: a tuple of containers we might own. used when we were
// provided with an rval at zip time. allows us to destroy the
// data when we're actually done iterating.
template <
typename StoreT,
typename BeginT,
typename EndT,
typename I = std::make_index_sequence<std::tuple_size_v<StoreT>>
>
class collection;
iterator //---------------------------------------------------------------------
end (void) template <
{ typename StoreT,
return iterator { { std::get<I> (m_end)... } }; typename BeginT,
} typename EndT,
std::size_t ...I
>
class collection<
StoreT,
BeginT,
EndT,
std::index_sequence<I...>
> {
public:
collection (StoreT _store, BeginT _begin, EndT _end):
m_store { std::move (_store) },
m_begin (std::move (_begin)),
m_end (std::move (_end))
{ ; }
auto begin (void)& { return iterator<BeginT> { m_begin }; }
auto end (void)& { return iterator<EndT> { m_end }; }
private: private:
IteratorT m_begin; StoreT m_store;
IteratorT m_end; BeginT m_begin;
const StoreT m_store; EndT m_end;
}; };
} }
@ -408,19 +410,16 @@ namespace util {
auto auto
zip (ContainerT&&... data) zip (ContainerT&&... data)
{ {
using IteratorT = std::tuple<decltype(std::begin(data))...>;
auto store = util::variadic::filter<std::is_rvalue_reference> (std::forward<ContainerT> (data)...);
return detail::zip::collection< return detail::zip::collection<
IteratorT, decltype (std::forward_as_tuple (data...)),
decltype(store), decltype (std::make_tuple (std::begin (data)...)),
decltype (std::make_tuple (std::end (data)...)),
std::make_index_sequence<sizeof...(ContainerT)> std::make_index_sequence<sizeof...(ContainerT)>
> { > (
std::forward_as_tuple (data...),
std::make_tuple (std::begin (data)...), std::make_tuple (std::begin (data)...),
std::make_tuple (std::end (data)...), std::make_tuple (std::end (data)...)
std::move (store) );
};
}; };

View File

@ -12,12 +12,17 @@ main (int, char**)
{ {
util::TAP::logger tap; util::TAP::logger tap;
// test that iteration across disparate types works as expected.
//
// the char array is an important element because it tends to decay to a
// pointer type unless we're paying careful attention.
{
std::vector<int> v_int { 1, 2, 3 }; std::vector<int> v_int { 1, 2, 3 };
std::array<float,3> a_float { 1.1f, 2.2f, 3.3f }; std::array<float,3> a_float { 1.1f, 2.2f, 3.3f };
char c_char[] = { 'a', 'b', 'c' }; char c_char[] = { '\0', 'b', 'c' };
bool success = true; bool success = true;
for (const auto &[i, v, a, c]: util::izip (v_int, a_float, c_char)) { for (auto [i, v, a, c]: util::izip (v_int, a_float, c_char)) {
success = success && success = success &&
v_int[i] == v && v_int[i] == v &&
util::equal (a_float[i], a) && util::equal (a_float[i], a) &&
@ -25,5 +30,24 @@ main (int, char**)
} }
tap.expect (success, "izip containers of int, float, and char and an initialiser_list"); tap.expect (success, "izip containers of int, float, and char and an initialiser_list");
}
// test that the obvious structured binding syntax gives references to
// the underlying values rather than silently supplying lvalues.
//
// we deliberately do not use any type of references in the range loop so
// that we check the syntax a user is likely to employ actually works.
// references may not be an obvious consideration.
{
const std::array<unsigned,3> src { 0, 1, 2 };
std::array<unsigned,3> dst { 2, 0, 1 };
for (auto [a,b]: util::zip (src, dst))
b = a;
tap.expect_eq (src, dst, "copy using structured bindings");
}
return tap.status (); return tap.status ();
} }

View File

@ -30,8 +30,9 @@ main (int, char**)
}; };
util::view src { csv.c_str (), csv.size () }; util::view src { csv.c_str (), csv.size () };
for (const auto &[tok, expected]: util::zip (util::tokeniser (src, ','), TESTS)) for (auto &&[tok, expected]: util::zip (util::tokeniser (src, ','), TESTS)) {
tap.expect (equal (tok, expected.value), "%s", expected.message); tap.expect (equal (tok, expected.value), "%s", expected.message);
}
return tap.status (); return tap.status ();
} }