Разделы публикаций

 
     
 

Построение дерева из списка ID-Parent

nectarin
228
    nectarin nectarin
Сергей NectarIn Новиков
Специализации: 1

Поскольку данная задача возникает у разработчиков довольно часто, хотел бы поделиться простым и эффективным решением. Всё, что нам потребуется — это исходный массив примерно такого вида:
$listIdParent = array(
  => array('parent' => 0),
  => array('parent' => 1),
  => array('parent' => 1),
  => array('parent' => 2),
  => array('parent' => 4)
);


Как нетрудно догадаться, ключи исходного списка являются ID, а поля parent содержат ссылки на родительский элемент в дереве. Элемент с parent, равным 0, является корневым. Такой элемент должен быть единственным в списке, и в целом дерево должно быть консистентным (не должно быть "битых" ссылок и рекурсивной вложенности) — и, пожалуй, это единственные ограничения. Таким образом, важной особенностью используемого метода является то, что элементы исходного списка могут идти в любом порядке, в отличие от известного способа с array_pop(), который требует, чтобы элементы были отсортированы по уровню (глубине вложенности).

Дерево строится следующей функцией:
/**
 * Построение дерева из списка ID-Parent.
 * 
 * @param array Исходный список.
 * @return array Дерево.
 */
function buildTree(array $listIdParent) {
  // Подготовка ID корневого узла.
  $rootId 0;
10   
11   // Обход списка и обработка узлов.
12   foreach ($listIdParent as $id => $node) {
13     if ($node['parent']) {
14       // Сохранение в родительском узле ссылки на текущий.
15       $listIdParent[$node['parent']]['sub'][$id] =& $listIdParent[$id];
16     } else {
17       // Сохранение ссылки на корневой элемент.
18       $rootId $id;
19     }
20   }
21   
22   // Возврат корневого узла, содержащего всё построенное дерево.
23   return array($rootId => $listIdParent[$rootId]);
24 }

Для повышения удобочитаемости в коде функции опущены необходимые проверки входного массива на корректность.

Результат работы функции выглядит так:
array(
  => array(
    'parent' => 0,
    'sub' => array(
      => array(
        'parent' => 1,
        'sub' => array(
          => array(
            'parent' => 2,
10             'sub' => array(
11               => array('parent' => 4)
12             )
13           )
14         )
15       ),
16       => array(
17         'parent' => 1
18       )
19     )
20   )
21 )

При сравнении с другими способами построения дерева из списка ID-Parent можно выделить следующие основные моменты:

  • отсутствие рекурсии; достаточно одной функции;

  • всего один цикл по исходному массиву для полного построения дерева;

  • все манипуляции с массивом заключаются в установке ссылок на дочерние элементы, что сильно экономит память и увеличивает быстродействие (по сравнению с методами, перемещающими сами элементы массива).


И хотя первые два пункта так же присущи алгоритму с array_pop(), третий пункт, а также меньшие ограничения на формат входного массива делают предложенное решение наиболее применимым в разработке.

3
Просмотров: 3616
     

Новости inPHP.org Новости inPHP.org
30.12.2011   С наступающим Новым Годом!
Команда inPHP.org поздравляет всех посетителей сайта проекта с наступающим Новым Годом! Вероятно, многие участники проекта заметили некоторое затишье в уходящем году. Поверьте, в следующем всё будет несколько иначе!
31.12.2010   С Новым Годом!
Уважаемые коллеги, коллектив проекта inphp.org искренне поздравляет вас с наступающим Новым Годом!
27.05.2010   Технические работы.
По техническим причинам возможны кратковременные перебои в работе сайта проекта в период с 27.05.2010 по 1.06.2010.
02.03.2010   Доступ участников к разработке тестирований.
В тестовом режиме запущен функционал, позволяющий участникам проекта разрабатывать собственные тестирования. Разработанные сообществом тестирования не имеют ограничений официальной аккредитации и в любое время доступны любому участнику.
 
Содействуют развитию Содействуют развитию
Участники проекта Участники проекта
suearth
60
    suearth suearth

Специализации: 0
poiuyt
120
    poiuyt poiuyt

Специализации: 0
tester
20
    tester tester

Специализации: 0
fiery_fenix
100
    fiery_fenix fiery_fenix
Вячеслав
Специализации: 0
kashukov
60
    kashukov kashukov

Специализации: 0
 
Статистика проекта Статистика проекта
Всего пользователей: 3648
Из них аккредитовано: 1468
Были сегодня: 2

УровеньПользователи
7742%
62376,5%
52075,7%
41945,3%
336510%
2401,1%
13509,6%
отсутствует218059,8%