Используем итераторы SPL. Часть 1.
Те, кто впервые сталкивается с термином "итерация", как правило обнаруживают в стандартной библиотеке PHP (далее по тексту SPL - Standart PHP Library) огромный список классов, которые имеют отношение к данному понятию. Такой объем информации может отпугнуть новичка, создавая впечатление, что итерация - это что-то из раздела программирования ради самого программирования.
Если вы активно используете PHP, то обязательно сталкиваетесь с обработкой массивов. А обработка массива - это, практически всегда, перебор его элементов. Даже бегло просмотрев на код PHP практически везде можно обнаружить цикл foreach
. Итерация - это процесс прохода по списку значений. С ее помощью можно обрабатывать объекты, массивы, директории и даже результат запроса к базе данных.
В первой части серии из двух уроков мы рассмотрим, что такое итерация и какие преимущества есть у классов из SPL. В SPL имеется большое количество различных итераторов, использование которых позволяет создавать более эффективный и легко читаемый код.
Зачем и когда использовать итераторы SPL
Обработка объектов итераторов очень похожа на обработку массивов. Многие начинающие программисты начинают с использования итераторов для массивов. Но реальные преимущества итераторы дают при переборе большого количества гораздо более сложных данных, чем простой массив.
Цикл foreach
делает копию массива, который ему передаётся. Если происходит обработка большого объема данных, то копирование массивов при каждом использовании цикла foreach
может быть нежелательным. Итераторы SPL инкапсулируют список и делают видимым один элемент в конкретный момент времени, что гораздо эффективнее.
При создании структур данных итераторы могут существенно повлиять на производительность, так как позволяют организовать отложенную загрузку данных. Отложенная загрузка дает возможность получать данные только тогда, когда они нужны. Также можно манипулировать данными (фильтровать, преобразовывать и так далее) перед передачей их пользователю.
Решение об использовании итераторов всегда остаётся за разработчиком. Итераторы имеют ряд преимуществ, но в некоторых случаях их применение может оказаться избыточным (например, для небольших наборов данных). Поэтому нужно принимать во внимание все факторы проекта.
Итерация массивов
Первый итератор, который мы рассмотрим - ArrayIterator
. Конструктор принимает массив в качестве параметра и предоставляет набор методов для его обработки. Например:
<?php // Массив (используем короткую нотацию PHP 5.4) $arr = ["sitepoint", "phpmaster", "buildmobile", "rubysource", "designfestival", "cloudspring"]; // Создаем ArrayIterator и передаем ему массив $iter = new ArrayIterator($arr); // Цикл для обработки объекта foreach ($iter as $key => $value) { echo $key . ": " . $value . "<br>"; }
Код выше приведенного примера выведет:
0: sitepoint 1: phpmaster 2: buildmobile 3: rubysource 4: designfestival 5: cloudspring
Обычно используется ArrayObject
, класс для обработки объектов как массивов в определенном контексте, вместо непосредственного применения ArrayIterator
. В данном случае автоматически создается ArrayIterator
когда используется цикл foreach
или можно вызвать метод ArrayIterator::getIterator()
.
Обратите внимание, что хотя ArrayObject
и ArrayIterator
ведут себя как массивы в данном контексте, они являются объектами. Попытка использовать встроенные функции для массивов, например, sort()
или array_keys()
, для них приведет к ошибке.
Использование ArrayIterator
ограничивается одномерными массивами. Тогда, когда можно обрабатывать многомерные массивы нужно использовать RecursiveArrayIterator
.
Обычный сценарий - связывание циклов foreach
или созадние рекурсивных функций для провреки всех элементов многомерного массива. Например:
<?php // Многомерный массив $arr = [ ["sitepoint", "phpmaster"], ["buildmobile", "rubysource"], ["designfestival", "cloudspring"], "not an array" ]; // Цикл по объекту foreach ($arr as $key => $value) { // Проверка на массив if (is_array($value)) { foreach ($value as $k => $v) { echo $k . ": " . $v . "<br>"; } } else { echo $key . ": " . $value . "<br>"; } }
Вывод пример будет иметь следующий вид:
0: sitepoint 1: phpmaster 0: buildmobile 1: rubysource 0: designfestival 1: cloudspring 3: not an array
Более элегантный подход реализуется с помощью RecursiveArrayIterator
.
<?php ... $iter = new RecursiveArrayIterator($arr); // Цикл по объекту // Нужно создать экземпляр RecursiveIteratorIterator foreach(new RecursiveIteratorIterator($iter) as $key => $value) { echo $key . ": " . $value . "<br>"; }
Данный код выведет то же, что и предыдущий пример.
Обратите внимание, что нужно создать ему новый экземпляр RecursiveIteratorIterator
и передать ему объект RecursiveArrayIterator
.
Итератор RecursiveArrayIterator
следует использовать при работе с многомерными массивами. RecursiveIteratorIterator
является декоратором, который выполняет всю работу. Он берет RecursiveArrayIterator
и проходит по любому элементу Iterable
, который найдет. По существу, декоратор "разглаживает" RecursiveArrayIterator
. Вы можете определять глубину обработки с помощью вызова RecursiveIteratorIterator::getDepth()
. Нужно осторожно применять RecursiveArrayIterator
и RecursiveIteratorIterator
при работе с объектами. Объекты будут распознаны как Iterable
и включены в цикл обработки.
Итерация директорий
Несомненно, что иногда вам приходится обрабатывать директории и их файлы. Существует множество методов для решения подобных задач с помощью встроенных функций PHP, например, scandir()
или glob()
. Но также можно использовать DirectoryIterator
. Даже в своей простой форме DirectoryIterator
- достаточно мощный инструмент. Но его можно также расширить.
Пример обработки директории с помощью DirectoryIterator
:
<?php // Создаем новый объект DirectoryIterator $dir = new DirectoryIterator("/my/directory/path"); // Цикл по содержанию директории foreach ($dir as $item) { echo $item . "<br>"; }
Вывод зависит от содержания директории, например, он может быть таким:
.. api index.php lib workspace
Для DirectoryIterator
, также как и для многих других итераторов SPL, можно использовать исключения для обработки ошибок.
<?php try { $dir = new DirectoryIterator("/non/existent/path"); foreach ($dir as $item) { echo $item . "<br>"; } } catch (Exception $e) { echo get_class($e) . ": " . $e->getMessage(); }
UnexpectedValueException: DirectoryIterator::__construct(/non/existent/path,/non/existent/path): The system cannot find the file specified. (code: 2)
С помощью набора различных методов, таких как DirectoryIterator::isDot()
, DirectoryIterator::getType()
и DirectoryIterator::getSize()
, можно получить практически любую информацию о директории. Также можно комбинировать DirectoryIterator
с FilterIterator
или RegexIterator
для получения списка файлов по определенным критериям. Например:
<?php class FileExtensionFilter extends FilterIterator { // Белый список расширений файлов protected $ext = ["php", "txt"]; // Абстрактный метод, который надо реализовать в подклассе public function accept() { return in_array($this->getExtension(), $this->ext); } } //Создаем новый итератор $dir = new FileExtensionFilter(new DirectoryIterator("./")); ...
В SPL также имеется RecursiveDirectoryIterator
, который можно использовать также, как и RecursiveArrayIterator
. Есть одна особенность. RecursiveDirectoryIterator
не возвращает пустых директорий: если директория содержит много поддиректорий, но без файлов, результат будет пустым.
<?php // Создаем новый объект RecursiveDirectoryIterator $iter = new RecursiveDirectoryIterator("/my/directory/path"); // Цикл по списку директории // Нужно создать новый экземпляр RecursiveIteratorIterator foreach (new RecursiveIteratorIterator($iter) as $item) { echo $item . "<br>"; }
Примерный вид результата работы кода:
./api/.htaccess ./api/index.php ./index.php ...
Заключение
В данном уроке представлены итераторы SPL, которые могут помочь существенно улучшить код. SPL предоставляет программисту большой набор инструментов для организации итераций. Иногда, классы могут показаться излишне обобщенными. В таких случаях можно легко добавить нужный функционал. В следующем уроке мы рассмотрим как сделать обработку пользовательского класса подобной обработке простого массива.
Источник: http://feedproxy.google.com/~r/ruseller/CdHX/~3/idUhTjQgGQo/lessons.php
Дайджест новых статей по интернет-маркетингу на ваш email
Новые статьи и публикации
- 2024-11-26 » Капитан грузового судна, или Как начать использовать Docker в своих проектах
- 2024-11-26 » Обеспечение безопасности ваших веб-приложений с помощью PHP OOP и PDO
- 2024-11-22 » Ошибки в Яндекс Вебмастере: как найти и исправить
- 2024-11-22 » Ошибки в Яндекс Вебмастере: как найти и исправить
- 2024-11-15 » Перенос сайта на WordPress с одного домена на другой
- 2024-11-08 » OSPanel 6: быстрый старт
- 2024-11-08 » Как установить PhpMyAdmin в Open Server Panel
- 2024-09-30 » Как быстро запустить Laravel на Windows
- 2024-09-25 » Next.js
- 2024-09-05 » OpenAI рассказал, как запретить ChatGPT использовать содержимое сайта для обучения
- 2024-08-28 » Чек-лист: как увеличить конверсию интернет-магазина на примере спортпита
- 2024-08-01 » WebSocket
- 2024-07-26 » Интеграция с Яндекс Еда
- 2024-07-26 » Интеграция с Эквайринг
- 2024-07-26 » Интеграция с СДЕК
- 2024-07-26 » Интеграция с Битрикс-24
- 2024-07-26 » Интеграция с Travelline
- 2024-07-26 » Интеграция с Iiko
- 2024-07-26 » Интеграция с Delivery Club
- 2024-07-26 » Интеграция с CRM
- 2024-07-26 » Интеграция с 1C-Бухгалтерия
- 2024-07-24 » Что такое сторителлинг: техники и примеры
- 2024-07-17 » Ошибка 404: что это такое и как ее использовать для бизнеса
- 2024-07-03 » Размещайте прайс-листы на FarPost.ru и продавайте товары быстро и выгодно
- 2024-07-01 » Профилирование кода в PHP
- 2024-06-28 » Изучаем ABC/XYZ-анализ: что это такое и какие решения с помощью него принимают
- 2024-06-17 » Зачем вам знать потребности клиента
- 2024-06-11 » Что нового в работе Яндекс Метрики: полный обзор обновления
- 2024-06-11 » Поведенческие факторы ранжирования в Яндексе
- 2024-06-11 » Скорость загрузки сайта: почему это важно и как влияет на ранжирование
Всегда храни верность своему начальнику - следующий, может быть еще хуже... |
Мы создаем сайты, которые работают! Профессионально обслуживаем и продвигаем их , а также по всей России и ближнему зарубежью с 2006 года!
Как мы работаем
Заявка
Позвоните или оставьте заявку на сайте.
Консультация
Обсуждаем что именно Вам нужно и помогаем определить как это лучше сделать!
Договор
Заключаем договор на оказание услуг, в котором прописаны условия и обязанности обеих сторон.
Выполнение работ
Непосредственно оказание требующихся услуг и работ по вашему заданию.
Поддержка
Сдача выполненых работ, последующие корректировки и поддержка при необходимости.