-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathtoken_iterator.hpp
116 lines (96 loc) · 4 KB
/
token_iterator.hpp
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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
// Boost token_iterator.hpp -------------------------------------------------//
// Copyright John R. Bandela 2001
// Permission to copy, use, modify, sell and distribute this software
// is granted provided this copyright notice appears in all
// copies. This software is provided "as is" without express or
// implied warranty, and with no claim as to its suitability for any
// purpose.
// See http://www.boost.org/libs/tokenizer for documentation.
#ifndef BOOST_TOKENIZER_POLICY_JRB051801_HPP_
#define BOOST_TOKENIZER_POLICY_JRB051801_HPP_
#include<boost/iterator_adaptors.hpp>
#include<boost/token_functions.hpp>
#include<utility>
#include<cassert>
namespace boost {
namespace detail{
// The base "iterator" for iterator adapter
template<class It>
class token_iterator_base
{
public:
std::pair<It,It> p_;
bool valid_;
token_iterator_base():p_(It(),It()),valid_(false){}
token_iterator_base(const It& b , const It& e )
:p_(b,e),valid_(false){}
operator It(){return p_.first;}
template<class T>
token_iterator_base(const token_iterator_base<T>& other)
:p_(other.p_),valid_(other.valid_){}
};
template<class Type, class TokenizerFunc>
class tokenizer_policy{
private:
TokenizerFunc func_;
Type tok_;
public:
tokenizer_policy(){}
tokenizer_policy(const TokenizerFunc& f):func_(f){};
template<class Base>
void initialize(Base& b){
if(b.valid_) return;
func_.reset();
b.valid_ = (b.p_.first != b.p_.second)?
func_(b.p_.first,b.p_.second,tok_):false;
}
template<class Iterator1, class Iterator2>
bool equal(const Iterator1& a, const Iterator2& b) const{
return (a.base().valid_ && b.base().valid_)
?(a.base().p_==b.base().p_)
:(a.base().valid_==b.base().valid_);
}
template<class Iterator>
typename Iterator::reference
dereference(const Iterator& a) const{
using namespace std;
assert(a.base().valid_);
return tok_;
}
template <class Iterator>
void increment(Iterator& b){
using namespace std;
assert(b.base().valid_);
b.base().valid_ = func_(b.base().p_.first,b.base().p_.second,tok_);
}
};
} // namespace detail
template <
class TokenizerFunc = char_delimiters_separator<char>,
class Iterator = std::string::const_iterator,
class Type = std::string
>
class token_iterator_generator {
private:
typedef Type value_type;
typedef detail::tokenizer_policy<Type, TokenizerFunc> policies;
typedef detail::token_iterator_base<Iterator> base;
typedef typename boost::detail::non_bidirectional_category<
Iterator>::type category;
public:
typedef boost::iterator_adaptor<base,policies,value_type,
const value_type&,const value_type*,category,std::ptrdiff_t> type;
};
// Type has to be first because it needs to be explicitly specified
// because there is no way the function can deduce it.
template<class Type, class Iterator, class TokenizerFunc>
typename token_iterator_generator<TokenizerFunc,Iterator,Type>::type
make_token_iterator(Iterator begin, Iterator end,const TokenizerFunc& fun){
typedef typename
token_iterator_generator<TokenizerFunc,Iterator,Type>::type ret_type;
detail::token_iterator_base<Iterator> b(begin,end);
detail::tokenizer_policy<Type,TokenizerFunc> f(fun);
return ret_type(b,f);
}
} // namespace boost
#endif