2003-11-19 06:18:38 +08:00
|
|
|
<?php
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** @file cachingiterator.inc
|
2004-11-01 03:05:37 +08:00
|
|
|
* @ingroup SPL
|
2004-10-30 04:58:58 +08:00
|
|
|
* @brief class CachingIterator
|
|
|
|
* @author Marcus Boerger
|
2005-02-09 03:10:06 +08:00
|
|
|
* @date 2003 - 2005
|
2004-10-30 04:58:58 +08:00
|
|
|
*
|
|
|
|
* SPL - Standard PHP Library
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2004-11-01 03:05:37 +08:00
|
|
|
* @brief Cached iteration over another Iterator
|
2004-10-30 04:58:58 +08:00
|
|
|
* @author Marcus Boerger
|
2005-09-15 11:33:04 +08:00
|
|
|
* @version 1.2
|
2005-02-17 08:24:57 +08:00
|
|
|
* @since PHP 5.0
|
2004-10-30 04:58:58 +08:00
|
|
|
*
|
2004-11-01 03:05:37 +08:00
|
|
|
* This iterator wrapper does a one ahead iteration. This way it knows whether
|
|
|
|
* the inner iterator has one more element.
|
|
|
|
*
|
|
|
|
* @note If you want to convert the elements into strings and the inner
|
|
|
|
* Iterator is an internal Iterator then you need to provide the
|
2005-09-15 11:33:04 +08:00
|
|
|
* flag CALL_TOSTRING to do the conversion when the actual element
|
2004-11-01 03:05:37 +08:00
|
|
|
* is being fetched. Otherwise the conversion would happen with the
|
|
|
|
* already changed iterator. If you do not need this then it you should
|
|
|
|
* omit this flag because it costs unneccessary work and time.
|
2004-10-30 04:58:58 +08:00
|
|
|
*/
|
2004-10-30 04:12:57 +08:00
|
|
|
class CachingIterator implements OuterIterator
|
2003-11-19 06:18:38 +08:00
|
|
|
{
|
2005-10-09 03:09:58 +08:00
|
|
|
const CALL_TOSTRING = 0x00000001;
|
|
|
|
const CATCH_GET_CHILD = 0x00000002;
|
|
|
|
const TOSTRING_USE_KEY = 0x00000010;
|
|
|
|
const TOSTRING_USE_CURRENT = 0x00000020;
|
2005-09-15 11:33:04 +08:00
|
|
|
|
2005-02-09 03:00:19 +08:00
|
|
|
private $it;
|
|
|
|
private $current;
|
|
|
|
private $key;
|
|
|
|
private $valid;
|
|
|
|
private $strValue;
|
2003-11-19 06:18:38 +08:00
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** Construct from another iterator
|
|
|
|
*
|
|
|
|
* @param it Iterator to cache
|
|
|
|
* @param flags Bitmask:
|
2005-09-15 11:33:04 +08:00
|
|
|
* - CALL_TOSTRING (whether to call __toString() for every element)
|
2004-10-30 04:58:58 +08:00
|
|
|
*/
|
2005-09-15 11:33:04 +08:00
|
|
|
function __construct(Iterator $it, $flags = self::CALL_TOSTRING)
|
2003-11-23 04:51:15 +08:00
|
|
|
{
|
2005-10-04 00:05:08 +08:00
|
|
|
if ((($flags & self::CALL_TOSTRING) && ($flags & (self::TOSTRING_USE_KEY|self::TOSTRING_USE_CURRENT)))
|
|
|
|
|| ((flags & (self::CIT_TOSTRING_USE_KEY|self::CIT_TOSTRING_USE_CURRENT)) == (self::CIT_TOSTRING_USE_KEY|self::CIT_TOSTRING_USE_CURRENT)))
|
|
|
|
{
|
|
|
|
throw new InvalidArgumentException('Flags must contain only one of CIT_CALL_TOSTRING, CIT_TOSTRING_USE_KEY, CIT_TOSTRING_USE_CURRENT');
|
|
|
|
}
|
2003-11-19 06:18:38 +08:00
|
|
|
$this->it = $it;
|
2005-10-04 00:05:08 +08:00
|
|
|
$this->flags = $flags & (0x0000FFFF);
|
2004-05-03 06:07:32 +08:00
|
|
|
$this->next();
|
2003-11-19 06:18:38 +08:00
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** Rewind the Iterator
|
|
|
|
*/
|
2003-11-23 04:51:15 +08:00
|
|
|
function rewind()
|
|
|
|
{
|
2003-11-19 06:18:38 +08:00
|
|
|
$this->it->rewind();
|
|
|
|
$this->next();
|
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** Forward to the next element
|
|
|
|
*/
|
2003-11-23 04:51:15 +08:00
|
|
|
function next()
|
|
|
|
{
|
2004-05-03 06:07:32 +08:00
|
|
|
if ($this->valid = $this->it->valid()) {
|
2003-11-19 06:18:38 +08:00
|
|
|
$this->current = $this->it->current();
|
|
|
|
$this->key = $this->it->key();
|
2005-09-15 11:33:04 +08:00
|
|
|
if ($this->flags & self::CALL_TOSTRING) {
|
2003-12-07 03:03:17 +08:00
|
|
|
if (is_object($this->current)) {
|
|
|
|
$this->strValue = $this->current->__toString();
|
|
|
|
} else {
|
|
|
|
$this->strValue = (string)$this->current;
|
|
|
|
}
|
2003-12-05 04:56:32 +08:00
|
|
|
}
|
2003-11-19 06:18:38 +08:00
|
|
|
} else {
|
|
|
|
$this->current = NULL;
|
|
|
|
$this->key = NULL;
|
2003-12-07 23:03:11 +08:00
|
|
|
$this->strValue = NULL;
|
2003-11-19 06:18:38 +08:00
|
|
|
}
|
|
|
|
$this->it->next();
|
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** @return whether teh iterator is valid
|
|
|
|
*/
|
2004-03-09 01:33:31 +08:00
|
|
|
function valid()
|
2003-11-23 04:51:15 +08:00
|
|
|
{
|
2004-05-03 06:07:32 +08:00
|
|
|
return $this->valid;
|
2003-11-19 06:18:38 +08:00
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** @return whether there is one more element
|
|
|
|
*/
|
2003-11-23 04:51:15 +08:00
|
|
|
function hasNext()
|
|
|
|
{
|
2004-03-09 01:33:31 +08:00
|
|
|
return $this->it->valid();
|
2003-11-19 06:18:38 +08:00
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** @return the current element
|
|
|
|
*/
|
2003-11-23 04:51:15 +08:00
|
|
|
function current()
|
|
|
|
{
|
2003-11-19 06:18:38 +08:00
|
|
|
return $this->current;
|
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** @return the current key
|
|
|
|
*/
|
2003-11-23 04:51:15 +08:00
|
|
|
function key()
|
|
|
|
{
|
2003-11-19 06:18:38 +08:00
|
|
|
return $this->key;
|
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** Aggregate the inner iterator
|
2004-11-01 03:05:37 +08:00
|
|
|
*
|
|
|
|
* @param func Name of method to invoke
|
|
|
|
* @param params Array of parameters to pass to method
|
2004-10-30 04:58:58 +08:00
|
|
|
*/
|
2003-11-23 04:51:15 +08:00
|
|
|
function __call($func, $params)
|
|
|
|
{
|
2003-11-19 06:18:38 +08:00
|
|
|
return call_user_func_array(array($this->it, $func), $params);
|
|
|
|
}
|
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/** @return the string represenatation that was generated for the current
|
|
|
|
* element
|
2005-09-15 11:33:04 +08:00
|
|
|
* @throw exception when CALL_TOSTRING was not specified in constructor
|
2004-10-30 04:58:58 +08:00
|
|
|
*/
|
2003-11-23 04:51:15 +08:00
|
|
|
function __toString()
|
|
|
|
{
|
2005-10-04 00:05:08 +08:00
|
|
|
if ($this->flags & self::TOSTRING_USE_KEY)
|
|
|
|
{
|
|
|
|
return $this->key;
|
|
|
|
}
|
|
|
|
else if ($this->flags & self::TOSTRING_USE_CURRENT)
|
|
|
|
{
|
|
|
|
return $this->current;
|
|
|
|
}
|
|
|
|
if (!$this->flags & self::CALL_TOSTRING)
|
|
|
|
{
|
2003-12-07 03:03:17 +08:00
|
|
|
throw new exception('CachingIterator does not fetch string value (see CachingIterator::__construct)');
|
|
|
|
}
|
|
|
|
return $this->strValue;
|
2003-11-19 06:18:38 +08:00
|
|
|
}
|
2004-10-30 04:12:57 +08:00
|
|
|
|
2004-10-30 04:58:58 +08:00
|
|
|
/**
|
|
|
|
* @return The inner iterator
|
|
|
|
*/
|
2004-10-30 04:12:57 +08:00
|
|
|
function getInnerIterator()
|
|
|
|
{
|
|
|
|
return $this->it;
|
|
|
|
}
|
2003-11-19 06:18:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
?>
|