В предыдущей статье была приведена одна из реализаций перевода n-мерного массива в строку. Так называемая - упаковка массива.
Ниже приведена реализация обратного хода - распаковки строки в массив.
Вопрос распаковки в отличии от упаковки сложнее, так как необходимо создать массив и учитывать все вложенные массивы.
Свое решение реализовал с использованием ссылки на массив и условной перезаписи, точнее - переопределении и принципах заполнения массивов. Держался этого хода решения изначально в связи с возможностью использовать единый способ записи ключей и соответствующих им значений в массив. Это своего рода - аналогия с интерфейсом доступа к массиву. Ссылка для скачивания приведенного листинга - download.
<?php ## Распаковка строкового представления в n-мерный массив
// Счетчик вложенности массивов
$in = 0;
// Используем для отлова элементов (ключ => значение)
$catched = 0;
// Строковое представление массива (упакованный вид)
$str = "[|first|element|second|element|[|first_1|element_1|second_1|element_1|third_1|element_1|therd_1|element_1|]|third|element|[|first_2|element_2|second_2|element_2|]|2|end|]";
// Перекидываем элементы из строки $str в список - массив $view
$view = explode("|", $str);
// Создаем пустой рабочий массив для распаковывания строкового представления //массива
$restore = array();
// Создаем рабочую ссылку на рабочий массив $restore.
$temp = &$restore;
// Проccматриваем массив $view
foreach($view as $val) {
// Открываем массив
if($val == "[") {
// Увеличиваем счетчик вложенности массивов
$in++;
// Перезаписываем рабочую ссылку на последующий элемент массива $restore
// Такого рода перезаписи используем для создания единого интерфейса доступа к //рабочему массиву
$temp = &$temp[];
}
// Закрываем массив
elseif($val == "]") {
// Используем счетчик вложенности массивов $in для переопределения ссылки $temp //на рабочий массив $restore
$step = $in - 1;
// Перезаписываем ссылку. По-сути, возврат на начало
$temp = &$restore;
// Устанавливаем ссылку на соотвествующий элемент массива $restore
// Такого рода перезаписи используем для создания единого интерфейса доступа к //рабочему массиву
while($step) {
$temp = &$temp[0];
$step--;
}
// Понижаем счетчик вложенности
$in--;
}
// Запись извлекаемых ключей и соответствующих значений в рабочий массив $restore //с помощью ссылки $temp.
// По-сути, $temp - интерфейс доступа
else {
// счетчик отлова очередного элемента массива $view
$catched++;
// Временный буфер для отлова ключей и соответствующих значений
$buffer[] = $val;
// Так как элементы в проссматриваемом массиве $view содержат ключ и значение в //паре, используем такой мини-фильтр
if($catched == "2") {
// Распределяем отловленный ключ и соответствющее значение в буферные //переменные $key и $value
list($key ,$value) = $buffer;
// Записываем ключ и соответствующее значенее в рабочий массив $restore
$temp[$key] = $value;
// Сброс счетчика отлова очередного элемента массива $view
$catched = 0;
// Очищаем временный буфер $buffer
unset($buffer);
}
}
}
// Так как изначально для распаковки был создан массив $restore, а первый и последний //элементы Sstr порождали новый массив $restore[0] = array(), куда фактически
// все и распаковывалось, то для идеальности результата необходимо их перезаписать
$restore = $restore[0];
// Смотрим на результат
print_r($restore);
?>
Да, решение данного вида не идеально, не эстетично, поэтому-то необходимо для этих - и не только - целей использовать стандартные методы сериализации PHP5:
// Упаковывка не только массивов, но и чего-угодно
string serialize(mixed $obj);
// Распаковка в исходном виде сериализованного объекта
mixed unserialize(string $st);
Если в представленной реализации упаковки массива в строку были ограничения (критичность значения содержащего символ "|"), то прибегая к этим функциям можно этого не бояться. Что уж говорить, если данные методы сериализации активно используются при работе с базой данных, упаковывая и распаковывая различного рода объекты.