1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
|
#include <iterator>
#include <type_traits>
#include <stdexcept>
namespace debug
{
template < typename ITERATOR > struct range_checked_iterator
{
static_assert( std::is_convertible< typename std::iterator_traits<ITERATOR>::iterator_category,
std::bidirectional_iterator_tag >::value,
"the iterator being adapted must be at least a bidirectional iterator" ) ;
using iterator_category = std::bidirectional_iterator_tag ;
using value_type = typename std::iterator_traits<ITERATOR>::value_type ;
using difference_type = typename std::iterator_traits<ITERATOR>::difference_type ;
using pointer = typename std::iterator_traits<ITERATOR>::pointer ;
using reference = typename std::iterator_traits<ITERATOR>::reference ;
range_checked_iterator( ITERATOR current, ITERATOR begin, ITERATOR end )
: begin(begin), end(end), current(current) {}
reference operator*() const { return *current ; }
pointer operator->() const { return &**this ; }
range_checked_iterator& operator++() { ++throw_if(end) ; return *this ; }
range_checked_iterator operator++(int) { const auto prev(*this) ; ++*this ; return prev ; }
range_checked_iterator& operator--() { --throw_if(begin) ; return *this ; }
range_checked_iterator operator--(int) { const auto prev(*this) ; --*this ; return prev ; }
bool operator== ( const range_checked_iterator& that ) const { return current == that.current ; }
bool operator!= ( const range_checked_iterator& that ) const { return !( *this == that ) ; }
private:
const ITERATOR begin, end ;
ITERATOR current ;
ITERATOR& throw_if( ITERATOR x )
{
if( current == x ) throw std::out_of_range( "range checked iterator: out of range" ) ;
return current ;
}
};
}
template < typename SEQ >
auto range_checked_begin( SEQ& seq ) -> debug::range_checked_iterator< decltype( std::begin(seq) ) >
{ return { std::begin(seq), std::begin(seq), std::end(seq) } ; }
template < typename SEQ >
auto range_checked_end( SEQ& seq ) -> debug::range_checked_iterator< decltype( std::end(seq) ) >
{ return { std::end(seq), std::begin(seq), std::end(seq) } ; }
#include <vector>
#include <string>
#include <algorithm>
#include <iostream>
int main()
{
const int arr[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 } ;
std::copy( range_checked_begin(arr), range_checked_end(arr), std::ostream_iterator<int>( std::cout, " ") ) ;
std::cout << '\n' ;
std::vector<std::string> vec { "1.abc", "2.defg", "3.hijklm", "4.nopqrst" } ;
std::copy( range_checked_begin(vec), range_checked_end(vec), std::ostream_iterator<std::string>( std::cout, " ") ) ;
std::cout << '\n' ;
auto begin = range_checked_begin(vec), end = range_checked_end(vec) ;
using reverse_iterator = std::reverse_iterator< decltype(begin) > ;
auto rbegin = reverse_iterator(end), rend = reverse_iterator(begin) ;
std::copy( rbegin, rend, std::ostream_iterator<std::string>( std::cout, " ") ) ;
std::cout << '\n' ;
std::reverse( range_checked_begin(vec), range_checked_end(vec) ) ;
std::copy( range_checked_begin(vec), range_checked_end(vec), std::ostream_iterator<std::string>( std::cout, " ") ) ;
std::cout << '\n' ;
// these should throw
try { --range_checked_begin(arr) ; } catch( const std::exception& e ) { std::cout << e.what() << '\n' ; }
try { ++range_checked_end(vec) ; } catch( const std::exception& e ) { std::cout << e.what() << '\n' ; }
try { --rbegin ; } catch( const std::exception& e ) { std::cout << e.what() << '\n' ; }
try { ++rend ; } catch( const std::exception& e ) { std::cout << e.what() << '\n' ; }
}
| |