Используем итераторы SPL. Часть 2

В первой части серии были описаны некоторые стандартные итераторы SPL с примерами их использования. Но в практической деятельности очень часто функционала стандартных итераторов оказывается недостаточно. В таких случаях приходится создавать свои собственные итераторы, а SPL предоставляет интерфейсы для решения практических задач.

Для того, чтобы объект можно было перебирать в цикле foreach, PHP требует, чтобы он был экземпляром Traversable. Вы не можете реализовать данный интерфейс непосредственно (хотя его и можно использовать для проверок instaceof), а нужно будет создавать интерфейс Iterator или IteratorAggregate.

 

Iterator

Iterator позволяет создавать объекты, которые будут перебираться непосредственно или использоваться другими итераторами. Данный интерфейс имеет пять методов, которые нужно реализовать: rewind(), current(), key(), next(), и valid().

Предположим.. что у нас есть библиотека и мы хотим осуществлять перебор коллекции книг. Ниже приводится пример реализации Iterator:

<?php
class Library implements Iterator
{
    // Внутренний указатель на текущую позицию
    // в наборе данных
    protected $position = 0;

    // Массив названий книг в бибилиотеке
    protected $books = [
        "Professional PHP Programming",
        "Programming Perl",
        "A Byte of Python",
        "The Ruby Way"
    ];

    // Данный метод перемещает указатель назад к началу списка
    // для перезапуска перебора
    public function rewind() {
        echo "rewinding <br>";
        $this->position = 0;
    }

    // this method returns the value at the current
    // position of the dataset
    public function current() {
        echo "current <br>";
        return $this->books[$this->position];
    }

    // Данный метод возвращает текущее значение указателя
    public function key() {
        echo "key <br>";
        return $this->position;
    }

    // Данный метод перемещает указатель на следующий пункт
    // в наборе данных
    public function next() {
        echo "next <br>";
        ++$this->position;
    }

    // Данный метод возвращает логическое значение результата проверки 
    // соответствия набору данных по текущему указателю
    public function valid() {
        echo "valid <br>";
        return isset($this->books[$this->position]);
    }
}

$library = new Library();
foreach ($library as $key => $value) {
    echo $key . ": " . $value . "<br>";
}

Результат работы приведенного кода:

rewinding
valid
current
key
0: Professional PHP Programming
next
valid
current
key
1: Programming Perl
next
valid
current
key
2: A Byte of Python
next
valid
current
key
3: The Ruby Way
next
valid

Когда начинается перебор, PHP вызывает метод rewind() для перемещения указателя в начало набора данных. Затем осуществляется проверка правильности данных с помощью метода valid(). Если возвращаемое значение true, то далее вызывается метод current(), чтобы получить значение текущего указателя. Так как запрашивается $key для цикла, то PHP вызывает метод key() для получения текущего значения. PHP затем вызывает метод next() для перемещения указателя и метод valid() для проверки правильности данных. Цикл продолжается до тех пор, пока метод  valid() не вернет false.

Iterator дает возможность реализовать практически любой итератор с нуля. Конечно, выше приведенный пример очень простой и предназначен только для демонстрации методов. В идеале, с таким наборами данных лучше использовать IteratorAggregate.

 

IteratorAggregate

IteratorAggregate требует реализации единственного метода getIterator(). Данный метод требуется для возвращения внешнего итератора, который будет использоваться для перебора. Вот как будет выглядеть выше приведенный пример с использованием IteratorAggregate:

<?php
class Library implements IteratorAggregate
{
    protected $books = [
        "Professional PHP Programming",
        "Programming Perl",
        "A Byte of Python",
        "The Ruby Way"
    ];

    // Возвращает итератор для данных
    public function getIterator() {
        echo "getIterator <br>";
        return new ArrayIterator($this->books);
    }
}

$library = new Library();
foreach($library as $key => $value) {
    echo $key . ": " . $value . "<br>";
}

Выше приведенный код выдаст:

getIterator
0: Professional PHP Programming
1: Programming Perl
2: A Byte of Python
3: The Ruby Way

В начале перебора PHP вызывает getIterator(), который возвращает итератор (в данном случае - итератор SPL ArrayIterator). PHP затем выполняет все соответствующие вызовы для итератора.

Оба интерфейса обеспечивают одинаковый функционал, поэтому иногда трудно выбрать Iterator или IteratorAggregate. В общем случае Iterator следует использовать в тех случаях, когда требуется реализовать функционал итератора с нуля, потому что IteratorAggregate не подходит.

 

Резюме

Создавать собственные итераторы достаточно просто, а SPL предоставляет отличные средства для решения подобных задач.

Источник: http://feedproxy.google.com/~r/ruseller/CdHX/~3/YKHSJPYBiUc/lessons.php

Читать комменты и комментировать

Добавить комментарий / отзыв



Защитный код
Обновить

Используем итераторы SPL. Часть 2 | | 2012-06-19 11:57:45 | | Статьи Web-мастеру | | В первой части серии были описаны некоторые стандартные итераторы SPL с примерами их использования. Но в практической деятельности очень часто функционала стандартных итераторов оказывается | РэдЛайн, создание сайта, заказать сайт, разработка сайтов, реклама в Интернете, продвижение, маркетинговые исследования, дизайн студия, веб дизайн, раскрутка сайта, создать сайт компании, сделать сайт, создание сайтов, изготовление сайта, обслуживание сайтов, изготовление сайтов, заказать интернет сайт, создать сайт, изготовить сайт, разработка сайта, web студия, создание веб сайта, поддержка сайта, сайт на заказ, сопровождение сайта, дизайн сайта, сайт под ключ, заказ сайта, реклама сайта, хостинг, регистрация доменов, хабаровск, краснодар, москва, комсомольск |
 
Поделиться с друзьями: