Отображение категорий в неупорядоченном списке

Я столкнулся с проблемой сортировки и отображения категорий из базы данных в неупорядоченном списке.

Я написал этот код для выполнения задачи:

$roots = $GLOBALS['sql']->query("SELECT `id`,`name` FROM `category` WHERE `parent`='0'");
$out = '<ul>';
foreach($roots as $root){
$out .= '<li>'.$root['name'].'<ul>';
$downs = $GLOBALS['sql']->query("SELECT `id`,`name` FROM `category` WHERE `parent`='".$root['id']."'");
foreach($downs as $down){
$out .= '<li>'.$down['name'].'<ul>';
$bots = $GLOBALS['sql']->query("SELECT `name` FROM `category` WHERE `parent`='".$down['id']."'");
foreach($bots as $bot){
$out .= '<li>'.$bot['name'].'</li>';
}
$out .= '</ul></li>';
}
$out .= '</ul></li>';
}
$out .= '</ul>';
return str_replace('<ul></ul>', '', $out);

Это, кажется, очень долго, если есть много категорий. Есть ли более эффективный способ?
Или, вообще, обрабатывает ли массив php быстрее, чем запрос MySQL?

РЕДАКТИРОВАТЬ:
Я попытался объединить таблицы, и с помощью этого запроса я получил правильную таблицу.

SELECT
root.name  AS root_name,
down1.name AS down1_name,
down2.name AS down2_name
FROM
category AS root
LEFT JOIN category AS down1 ON down1.parent = root.id
LEFT JOIN category AS down2 ON down2.parent = down1.id
WHERE
root.parent = '0'
ORDER BY
root_name, down1_name, down2_name

Но как мне обработать эту таблицу в php, чтобы получить неупорядоченный список? и это еще быстрее?

0

Решение

Попробуйте что-то вроде этого:

$rows = $GLOBALS['sql']->query("SELECT
root.name AS root_name,
down1.name AS down1_name,
down2.name AS down2_name
FROM category AS root
LEFT JOIN category AS down1 ON down1.parent = root.id
LEFT JOIN category AS down2 ON down2.parent = down1.id
WHERE root.parent = '0'
ORDER BY root_name, down1_name, down2_name
");

$tree = array();
foreach ($rows as $row) {
$tree[$row['root_name']][$row['down1_name'] = $row['down2_name'];
}

$out = '';
foreach ($tree as $rootName => $root) {
$out .= "<li>{$rootName}<ul>";
foreach ($root as $down1Name => $down1) {
$out .= "<li>{$down1Name}<ul>";
foreach ($down1 as $down2Name) {
$out .= "<li>{$down2Name}</li>";
}
$out .= "</ul></li>";
}
$out .= "</ul></li>";
}

В первом цикле (foreach ($rows as $row)) вы строите древовидный массив. Во втором вложенном цикле вы строите свою HTML-структуру.

0

Другие решения

Попробуйте этот запрос:

select main_cat.name as main_category, sub_cat.name as sub_category, group_join(',', bot_cat.name) as bot_categories
from category main_cat
join category sub_cat
on (main_cat.parent = 0) and (sub_cat.parent = main.cat.id)
join category bot_cat
on bot_cat.parent = sub_cat.id
group by main_cat.name, sub_cat.name
order by main_cat.name, sub_cat.name

Это приведет к результирующему набору из трех столбцов:

  • main_category будет название основной категории
  • sub_category будет название подкатегории
  • bot_categories будет разделенным запятыми набором значений категорий ботов

Запрос сгруппирован по имени категории и имени подкатегории и упорядочен по этим значениям.

Использование PHP:

$previousCategory = null;
$categories = $GLOBALS['sql']->query("select main_cat.name as main_category, sub_cat.name as sub_category, group_join(',', bot_cat.name) as bot_categories
from category main_cat
join category sub_cat
on (main_cat.parent = 0) and (sub_cat.parent = main.cat.id)
join category bot_cat
on bot_cat.parent = sub_cat.id
group by main_cat.name, sub_cat.name
order by main_cat.name, sub_cat.name
");

foreach ($categories as $category) {
$newCat = ($category["main_category"] !== $previousCategory);
$previousCategory = $category["main_category"];
if ($newCat) {
//Handle new cat start
}
//Handle subcategory start ($category["sub_category"])
$bots = explode(',', $category['bot_categories']);
foreach ($bots as $bot) {
//Handle bot category
}
//Handle subcategory end ($category["sub_category"])
if ($newCat) {
//Handle new cat end
}
}

Объяснение:

Ваша программа работала медленно, потому что она должна была отправить запрос для корневых категорий, запрос для каждой корневой категории, чтобы получить их подкатегории, и запрос для каждой подкатегории, чтобы получить их категории ботов. Это привело к большому количеству запросов, отправленных в базу данных. Вместо этого вы можете решить свою проблему с помощью одного запроса.

0

По вопросам рекламы [email protected]