Задать вопрос
@pomozov

Как построить древовидный массив неограниченной вложенности?

Есть 2 массива. Первый массив это родители первого уровня, а второй потомки (родитель указан в ключе 'referer' ). Как их иерархически объединить в третий, при условии, что вложенность может быть условно неограниченной?

Массив 1 (родители первого уровня):

$array1 = [
    ['page'=>'1.ru', 'title'=>'—', 'childs'=>[]],
    ['page'=>'3.ru', 'title'=>'—', 'childs'=>[]],
    ['page'=>'6.ru', 'title'=>'—', 'childs'=>[]]
];

Массив 2 (потомки):

$array2 = [
    ['page'=>'666.ru', 'title'=>'+', 'referer'=>'66.ru'],
    ['page'=>'33.ru' , 'title'=>'+', 'referer'=>'3.ru'],
    ['page'=>'66.ru' , 'title'=>'+', 'referer'=>'6.ru']
];

Массив 3 (желаемый результат):

$array3 = [
    ['page'=>'1.ru', 'title'=>'—', 'childs'=>[]],
    ['page'=>'3.ru', 'title'=>'—', 'childs'=>[
        ['page'=>'33.ru' , 'title'=>'+', 'childs'=>[]],
    ]],
    ['page'=>'6.ru', 'title'=>'—', 'childs'=>[
        ['page'=>'66.ru' , 'title'=>'+', 'childs'=>[
            ['page'=>'666.ru', 'title'=>'+', 'childs'=>[]]
        ]]
    ]]
];
  • Вопрос задан
  • 391 просмотр
Подписаться 1 Простой Комментировать
Пригласить эксперта
Ответы на вопрос 3
Alexandroppolus
@Alexandroppolus
кодир
вложенность мотет быть неограниченной

выглядит так, будто у тебя бесконечно оперативы и ты хочешь куда-то её применить )))

Вот вариант за линейное время работы (из предположения, что хэш-таблица работает в среднем за О(1))
без рекурсии
на js, переведи в ПХП сам

код

const array1 = [
    {page: '1.ru', title: 'title 1'},
    {page: '3.ru', title: 'title 3'},
    {page: '6.ru', title: 'title 6'},
];

const array2 = [
    {page: '666.ru', title: 'title 666', referer: '66.ru'},
    {page: '33.ru', title: 'title 33', referer: '3.ru'},
    {page: '66.ru', title: 'title 66', referer: '6.ru'},
];

function createTree(roots, notRoots) {
    if (!roots || !roots.length) { return []; }

    const childsMap = new Map();

    // строим карту (page => список_чилдов), создавая узлы дерева для чилдов
    notRoots.forEach((child) => {
        let childs = childsMap.get(child.referer);

        if (!childs) {
            childs = [];
            childsMap.set(child.referer, childs);
        }

        childs.push({
            page: child.page,
            title: child.title,
        });
    });

    // всем чилдам (вновь созданным узлам), которые попали в карту, проставляем чилдов
    childsMap.forEach((childs) => {
        childs.forEach((child) => {
            child.childs = childsMap.get(child.page) || [];
        })
    });

    // обходим корневые элементы, создаем для них узлы дерева, подставляем чилды по карте
    return roots.map((root) => {
        return {
            page: root.page,
            title: root.title,
            childs: childsMap.get(root.page) || []
        };
    })
}

console.log(createTree(array1, array2));

Ответ написан
65536
@65536
Как

рекурсивно.

а связи-то у вас где хранятся? в изначальных массивах нужны идентификаторы и ссылки на родительские идентификаторы. потом надо проиндексировать их по этим ссылкам, то есть надо сделать массив где ключами будут иды, а элементами - массивы с элементами соответствующего родителя. потом рекурсивно идем сверху вниз, взяли первый уровень, листаем смотрим есть ли в том индексе элементы, ссылающиеся на текущий элемент, если есть, то погружаемся. перед подгружением нужно записать в результирующий массив данные по текущему пути. именно перед, потому что если после погружения, то вложенное затрётся. понадобится функция для записи в массывы по пути (типа такой https://laravel.su/docs/8.x/helpers#method-array-set). путём должно быть перечисление идов через спецсимвол. текущий путь можно хранить во внешней переменной, при погружении добавлять к нему сегмент с идом, в который погрузились, после погружения убирать его
Ответ написан
Комментировать
@wadowad
Вот такой говнокод нахрапом, но нужно добавить везде проверки:

$tree = $array1;

$array = $array2;

//вытаскивает значения из массива с детьми, и если место куда вставить ребёнка не нашлось, помещает его в конец
//Внимание! Функция войдёт в рекурсию, если места для ребёнка не найдётся - этот момент нужно фиксить
function one(array &$tree, array &$array) {
	//if (empty($array)) return false;

	$search = array_shift($array);

	$temp = $tree;

	$tree = two($tree, $search);
	if (serialize($tree) == serialize($temp)) $array[] = $search;

	if (!empty($array)) one($tree, $array);
}

//ищет куда вставить детей
function two(array $array, array $search) {
	$tree = [];

	foreach($array as $key => $value) {
		$tree[$key] = $value;

		if ($value['page'] == $search['referer']) {
			$tree[$key]['childs'][] = $search;
		} elseif (!empty($value['childs'])) {
			$tree[$key]['childs'] = two($array[$key]['childs'], $search);
		}
	}

	return $tree;
}

one($tree, $array);

var_dump($tree);
Ответ написан
Комментировать
Ваш ответ на вопрос

Войдите, чтобы написать ответ

Похожие вопросы