I am using PHP 5.3.6 from MAMP.
I have a use case where it would be best to use PHP’s Iterator interface methods, next(), current(), and valid() to iterate through a collection. A foreach loop will NOT work for me in my particular situation. A simplified while loop might look like
<?php
while ($iter->valid()) {
// do something with $iter->current()
$iter->next();
}
Should the above code always work when $iter implements PHP’s Iterator interface? How does PHP’s foreach keyword deal with Iterators?
The reason I ask is that the code I am writing may be given an ArrayIterator or a MongoCursor. Both implement PHP’s Iterator interface but they behave differently. I would like to know if there is a bug in PHP or PHP’s Mongo extension.
ArrayIterator::valid() returns true before any call to next() — immediately after the ArrayIterator is created.
MongoCursor::valid() only returns true after the first call to next(). Therefore the while loop above will never execute.
At risk of being verbose, the following code demonstrates these assertions:
<?php
// Set up array iterator
$arr = array("first");
$iter = new \ArrayIterator($arr);
// Test array iterator
echo(($iter->valid() ? "true" : "false")."\n"); // Echoes true
var_dump($iter->current()."\n"); // "first"
$iter->next();
echo(($iter->valid() ? "true" : "false")."\n"); // Echoes false
// Set up mongo iterator
$m = new \Mongo();
$collection = $m->selectDB("iterTest")->selectCollection("mystuff");
$collection->drop(); // Ensure collection is empty
$collection->insert(array('a' => 'b'));
$miter = $collection->find(); // should find one object
// Test mongo iterator
echo(($miter->valid() ? "true" : "false")."\n"); // Echoes false
$miter->next();
echo(($miter->valid() ? "true" : "false")."\n"); // Echoes true
var_dump($miter->current()); // Array(...)
Which implementation is correct? I found little documentation to support either behavior, and the official PHP documentation is either ambiguous or I’m reading it wrong. The doc for Iterator::valid() states:
This method is called after Iterator::rewind() and Iterator::next() to check if the current position is valid.
This would suggest that my while loop should first call next().
Yet the PHP documentation for Iterator::next states:
This method is called after each foreach loop.
This would suggest that my while loop is correct as written.
To summarize – how should PHP iterators behave?
This is an interesting question. I’m not sure why a
foreachwon’t work for you, but I have some ideas.Take a look at the example given on the Iterator interface reference page. It shows the order in which PHP’s internal implementation of
foreachcalls theIteratormethods. In particular, notice that when theforeachis first set up, the very first call is torewind(). This example, though it’s not well-annotated, is the basis for my answer.I’m not sure why a
MongoCursorwould not return true forvalid()until afternext()is called, but you should be able to reset either type of object by callingrewind()prior to your loop. So you would have:I believe this should work for you. If it does not, the Mongo class may have a bug in it.
Edit: Mike Purcell’s answer correctly calls out that ArrayIterator and Iterator are not the same. However, ArrayIterator implements Iterator, so you should be able to use
rewind()as I show above on either of them.