Hi Levi Morrison, > > Currently, there don't seem to be any internal classes that can be used to > > store a copy of the keys and values of an arbitrary Traversable. > > > > - This would help in eagerly evaluating the result of a generator in a > > memory efficient way that could be exactly stored and reused > > e.g. `return $this->cachedResults ??= new > >\RewindableKeyValueIterator($this->innerResultsGenerator());` > > https://externals.io/message/108767#108797 > > - This would be useful to exactly represent the keys of sequences with > > repeated keys (e.g. `yield 'first'; yield 'second';` implicitly uses the > > key `0` twice.) > > - This would be convenient to have to differentiate between 1, '1', and > > true. > > - This would be useful if php were to add internal global functions that > > act on iterables and return Traversables with potentially repeated keys > > based on those iterables, > > e.g. map(), filter(), take(), flip(), etc > > - If PHP were to add more iterable methods, being able to save an immutable > > copy of a traversable's result would be useful for end users. > > - If PHP were to add a Map (ordered hash map with null/any > > scalar/arrays/objects as keys) class type in the future, > > and that implemented IteratorAggregate, the return type of getIterator() > >would need something like RewindableKeyValueIterator. > > - The lack of a relevant datatype is among the reasons why classes such as > > SplObjectStorage are still an Iterator instead of an IteratorAggregate. > > (and backwards compatibility) > > > > ``` > > final class KeyValueSequenceIterator implements Iterator { > > // loop over all values in $values and store a copy, converting > > // references in top-level array values to non-references > > public function __construct(iterable $values) {...} > > public static function fromKeyValuePairs(iterable $entries): self {...} > >// fromKeyValuePairs([[$key1, $value1]]) > > public function rewind(): void {...} > > public function next(): void {...} > > public function current(): mixed {...} > > public function key(): mixed {...} > > public function valid(): bool {...} > > // and __debugInfo, __clone(), etc. > > } > > ``` > > The names `RewindableKeyValueIterator` and `KeyValueSequenceIterator` > are just long-form descriptions of the Iterator API. I don't think we > need such long names. The name should focus on what it brings. > > What it brings is a caching iterator around another iterator, that > includes re-windability. The SPL provides a `CachingIterator`, but I > assume it is inadequate somehow (it is basically SPL tradition). Can > you specifically discuss the shortcomings of `CachingIterator` and how > you will address them? To that end, have you implemented a > proof-of-concept for the proposed iterator?
It caches the results by coercing keys and inserting them into an array, which is a different use case. If you rewind a CachingIterator, it rewinds the iterator that it wraps, which throws for Generators and other types of iterators. For RewindableKeyValueIterator, it needs to store keys that can't be used as array keys. I was planning to implement a proof of concept if there wasn't widespread opposition and if nobody pointed out the functionality already existed. ``` <?php function dump_iterable(iterable $x) { foreach ($x as $key => $value) { printf("Key: %s\nValue: %s\n", json_encode($key), json_encode($value)); } } function yields_values(): Generator { yield 0 => 'first'; yield '0' => 'second'; } $c = new CachingIterator(yields_values(), CachingIterator::FULL_CACHE); echo "First CachingIterator iteration\n"; dump_iterable($c); var_export($c->getCache()); echo "\n"; // array(0 => 'second') does not represent that data echo "Second CachingIterator iteration\n"; // Fatal error: Uncaught Exception: Cannot rewind a generator that was already run dump_iterable($c); ``` Thanks, - Tyson -- PHP Internals - PHP Runtime Development Mailing List To unsubscribe, visit: https://www.php.net/unsub.php