-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcombinations.hpp
215 lines (202 loc) · 6.64 KB
/
combinations.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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
#ifndef COMBINATIONS_HPP
#define COMBINATIONS_HPP
#include <iterator>
#include <limits>
#include <numeric>
#include <vector>
// Precondition: all sets between iterators should be non-empty:
// contain at least 1 element
template <typename SetsIter>
class combinations
{
public:
typedef combinations<SetsIter> Combinations_type;
typedef typename std::iterator_traits<SetsIter>::value_type Set;
typedef typename std::vector<typename Set::const_iterator> Combination;
typedef Combination value_type;
typedef long long difference_type;
typedef size_t size_type;
class const_iterator
{
public:
typedef std::bidirectional_iterator_tag iterator_category;
typedef typename Combinations_type::difference_type difference_type;
typedef typename Combinations_type::value_type value_type;
typedef const Combination reference;
typedef const Combination pointer;
const_iterator() {}
const_iterator(const const_iterator& other)
: first_(other.first_)
, last_(other.last_)
, combination_(other.combination_)
{}
const_iterator(SetsIter first, SetsIter last, Combination combination)
: first_(first)
, last_(last)
, combination_(combination)
{}
static const_iterator
make_begin(const SetsIter first, const SetsIter last)
{
Combination combination;
for(SetsIter it = first; it != last; ++it)
combination.push_back(it->begin());
return const_iterator(first, last, combination);
}
static const_iterator
make_end(const SetsIter first, const SetsIter last)
{
SetsIter it = first;
Combination combination(1, it->end());
for(++it; it != last; ++it)
combination.push_back(--it->end());
return const_iterator(first, last, combination);
}
~const_iterator() {}
const_iterator& operator=(const const_iterator other)
{
swap(*this, other);
return *this;
}
bool operator==(const const_iterator& other) const
{
return first_ == other.first_ && last_ == other.last_ &&
combination_ == other.combination_;
}
bool operator!=(const const_iterator& other) const
{
return !(*this == other);
}
const_iterator& operator++()
{
typename Combination::iterator combIt = combination_.begin();
for(SetsIter it = first_; it != last_; ++it, ++combIt)
{
if(++(*combIt) != it->end())
return *this;
*combIt = it->begin();
}
set_to_end_();
return *this;
}
const_iterator& operator--()
{
typename Combination::iterator combIt = combination_.begin();
for(SetsIter it = first_; it != last_; ++it, ++combIt)
{
if(*combIt != it->begin())
{
--(*combIt);
return *this;
}
*combIt = --it->end();
}
set_to_begin_();
return *this;
}
Combination operator*() const { return combination_; }
Combination operator->() const { return combination_; }
friend void swap(
const_iterator& first,
const_iterator& second) // nothrow
{
std::swap(first.first_, second.first_);
std::swap(first.last_, second.last_);
std::swap(first.combination_, second.combination_);
}
void swap(Combinations_type other) { swap(*this, other); }
private:
void set_to_end_()
{
typename Combination::iterator combIt = combination_.begin();
SetsIter it = first_;
*(combIt++) = (it++)->end();
for(; it != last_; ++it, ++combIt)
*combIt = --it->end();
}
void set_to_begin_()
{
typename Combination::iterator combIt = combination_.begin();
for(SetsIter it = first_; it != last_; ++it, ++combIt)
*combIt = it->begin();
}
friend class combinations<SetsIter>;
private:
SetsIter first_;
SetsIter last_;
Combination combination_;
};
typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
combinations() {}
combinations(SetsIter first, SetsIter last)
: first_(first)
, last_(last)
{}
combinations(const combinations<SetsIter>& other)
: first_(other.first_)
, last_(other.last_)
{}
~combinations() {}
combinations& operator=(const combinations<SetsIter>& other)
{
swap(*this, other);
return *this;
}
bool operator==(const combinations<SetsIter>& other) const
{
return first_ == other.first_ && last_ == other.last_;
}
bool operator!=(const combinations<SetsIter>& other) const
{
return !(*this == other);
}
const_iterator cbegin() const
{
return const_iterator::make_begin(first_, last_);
}
const_iterator begin() const { return cbegin(); }
const_iterator cend() const
{
return const_iterator::make_end(first_, last_);
}
const_iterator end() const { return cend(); }
const_reverse_iterator crbegin() const
{
return const_reverse_iterator(end());
}
const_reverse_iterator rbegin() const { return crbegin(); }
const_reverse_iterator crend() const
{
return const_reverse_iterator(begin());
}
const_reverse_iterator rend() const { return crend(); }
friend void swap(
combinations<SetsIter>& first,
combinations<SetsIter>& second) // nothrow
{
std::swap(first.first_, second.first_);
std::swap(first.last_, second.last_);
}
void swap(combinations<SetsIter> other) { swap(*this, other); }
size_type size() const
{
return std::accumulate(first_, last_, 1, mult_by_set_size_);
}
size_type max_size() const { return std::numeric_limits<size_type>::max(); }
bool empty() const { return first_ == last_; }
private:
static size_type mult_by_set_size_(const size_type prev_sz, const Set& set)
{
return prev_sz * set.size();
}
private:
SetsIter first_;
SetsIter last_;
};
template <typename Sets>
combinations<typename Sets::const_iterator> make_combinations(const Sets& data)
{
return combinations<typename Sets::const_iterator>(
data.cbegin(), data.cend());
}
#endif // COMBINATIONS_HPP