-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSplitIterator.php
227 lines (213 loc) · 7.34 KB
/
SplitIterator.php
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
216
217
218
219
220
221
222
223
224
225
226
227
<?php
require_once("SplitIterator/SplitInnerIterator.php");
/**
* An {@see IteratorIterator} that splits another iterator ({@see
* Traversable}) into multiple iterators.
*
* This is a template (an abstract base class) for generic splitting
* iterators: You just need to subclass it and implement the method
* {@see needsSplit()} that decides on where the inner iterator should
* be split up.
*
* See {@see SplitCallbackIterator} for an alternative implementation,
* in case you don't want to subclass, but rather just hand in a
* callback to do the split decision in a ready-made class.
*
* A minimal example for a splitting iterator that splits any other
* iterator with numeric keys (at each key position that divides by
* three without remainder):
*
* <code>
* // Inherit from SplitIterator
* class SplitAtKeyDivisibleByThree extends SplitIterator {
*
* // Just add the split decision: It gets called with each
* // item's key and value -- a split takes place if this
* // method returns TRUE
* public function needsSplit($key, $item) {
* // Divide the key by three (modulus), and if there's no
* // remainder, split!
* return $key % 3 == 0;
* }
* }
* </code>
*
* Let's apply this trivial example to an Iterator that returns the
* letters 'a' to 'z' with the keys 1 to 26:
*
* <code>
* // Quick way to construct an iterator over 1..26 => 'a'..'z'
* $testArray = array_combine(range(1, 26), range('a', 'z'));
* $testIterator = new ArrayIterator($testArray);
*
* // Now apply our splitter to it
* $splitter = new SplitAtKeyDivisibleByThree($testIterator);
*
* // And we have the letters arranged into groups of three,
* // so the following prints:
* // abc
* // def
* // ghi
* // ...
* foreach($splitter as $group) {
* foreach($group as $letter)
* print $letter;
* print "\n";
* }
* </code>
*
* The inner iterator is in fact not really "physically" split: This
* class just generates a sort of dynamic "view" on top of it. The
* advantage of this being that no real additional memory overhead is
* created, even if the items in the inner iterator are large. A
* possible disadvantage is, that the {@see SplitInnerIterator}s
* returned for the individual splits can't be rewound and only one of
* them can be iterated over at any given time (once you retrieve the
* next one of the split-off iterators, the last one is invalidated).
* This is due to the fact that jumping back and forward between
* individual splits, and rewinding them alone, would actually mean to
* seek randomly through the inner iterator.
*
* For most applications (that resemble two nested loops, as in the
* example above) these restrictions however do not matter. If they
* do, you might want to just override {@see current()} and copy the
* contents of the {@see SplitInnerIterator} into another type of
* iterator that can live independently.
*
* @author Beat Vontobel
* @since 2015-05-20
* @package MeteoNews\phplib
* @subpackage Iterators
*
* @see SplitCallbackIterator SplitInnerIterator
*/
abstract class SplitIterator extends IteratorIterator {
/**
* Flag to denote that splitting should take place after the element
* that fired on {@see needsSplit()} (i.e. this very element will
* still be returned from the previous split-off iterator).
*
* @see SPLIT_BEFORE
*/
const SPLIT_AFTER = 0;
/**
* Flag to denote that splitting should take place before the
* element that fired on {@see needsSplit()} (i.e. this very element
* will already be returned by the next split-off iterator).
*
* @see SPLIT_AFTER
*/
const SPLIT_BEFORE = 1;
/**
* Flag to denote the time of splitting with regards to {@see
* needsSplit()} firing.
*
* @see SPLIT_AFTER, SPLIT_BEFORE
*/
protected $splitWhen = self::SPLIT_AFTER;
/**
* The current item (which is the split-off iterator) returned from
* this class.
*
* @val NULL|SplitInnerIterator
*/
private $currentSplitIterator;
/**
* The key for the current item returned by this class. We generate
* them as integer indices, starting at 0.
*
* @val int
*/
private $key = 0;
/**
* Decide if it's necessary to split-off a new iterator.
*
* Every child needs to implement this method: It is called for each
* item returned from the inner iterator, with that item's key and
* value. The splitting iterator needs to return TRUE if this item
* should split-off a new iterator.
*
* @param scalar The key of the current item as returned by the
* inner iterator.
*
* @param mixed The current value as returned by the inner iterator.
*
* @return bool TRUE if a new split-off iterator should be started
* for the current item returned from the inner iterator.
*/
abstract public function needsSplit($key, $current);
/**
* The {@see SplitIterator} returns generated integer keys for the
* split-off iterators it creates.
*
* The index starts at 0 for the first iterator returned.
*
* @return int
*/
public function key() {
return $this->key;
}
/**
* Returns the currently active split-off iterator.
*
* @return SplitInnerIterator
*/
public function current() {
return $this->currentSplitIterator;
}
public function next() {
// In case we get called while one of our split iterators
// is still active (early break): Eat that one first
while(isset($this->currentSplitIterator)
&& $this->currentSplitIterator->valid())
$this->currentSplitIterator->next();
if($this->splitOff())
$this->key++;
}
public function rewind() {
parent::rewind();
$this->key = 0;
$this->splitOff();
}
public function valid() {
return isset($this->currentSplitIterator);
}
/**
* Splits off a new {@see SplitInnerIterator} and makes it our
* current item.
*
* Invalidates a possibly earlier created {@see SplitInnerIterator},
* and checks if we really have more items available before
* splitting off.
*
* This internal method exists as the exact same code flow is needed
* in {@see rewind()} as well as {@see next()}.
*
* @return bool TRUE if a new {@see SplitInnerIterator} could be
* split off, FALSE otherwise.
*/
private function splitOff() {
if(isset($this->currentSplitIterator)) {
$this->currentSplitIterator->invalidate();
$this->currentSplitIterator = NULL;
}
// INVESTIGATE: We had parent::valid() here before, which should
// actually end up at the same code as
// $this->getInnerIterator()->valid(), but doesn't for some
// reason. Maybe some form of PHP internal cache? Or other
// mess-up in the IteratorIterator? Or did I really loose
// my mind?
if($this->getInnerIterator()->valid()) {
$this->currentSplitIterator
= new SplitInnerIterator($this->getInnerIterator(),
array($this, 'needsSplit'),
$this->splitWhen == self::SPLIT_AFTER
? SplitInnerIterator::STOP_AFTER
: SplitInnerIterator::STOP_BEFORE);
return TRUE;
} else {
return FALSE;
}
}
}
?>