SQL Server CTE Recursion - отслеживание дерева на определенном уровне дерева для каждого узла
У меня есть две таблицы: Item (ItemID, CatID, Name) и Category (CatID, ParentID, Name, CatLevel). Категория - это древовидная иерархия категорий, в которой могут быть элементы. Таблица также отслеживает, на каком уровне категории находится каждый узел. Вот пример:
1. Animal (CatLevel = 1)
2. Dog (CatLevel = 2)
3. Beagle (CatLevel = 3)
4. Labrador (CatLevel = 3)
5. Yellow (CatLevel = 4)
6. Chocolate (CatLevel = 4)
7. Black (CatLevel = 4)
8. Cat (CatLevel = 2)
9. Tabby (CatLevel = 3)
10. Horse (CatLevel = 2)
11. Stallion (CatLevel = 3)
Я пытаюсь найти запрос, который дает мне список элементов с их обычным CatID и CatID на уровне, который я указываю (например, уровень 2). Пример вывода:
Item CatID LevelTwoCatID
______ _____ _____________
Dog 2 2
Beagle 3 2
Labrador 4 2
Yellow 5 2
Chocolate 6 2
Black 7 2
Cat 8 8
Tabby 9 8
Horse 10 10
Stallion 11 10
Я знаю, как отследить до категории отдельного узла, но я не знаю, как сделать более общий запрос, который предоставит "LevelTwoCatID" для каждого узла. Вот как я это сделаю для одного:
WITH Tree AS
(
SELECT * FROM Category WHERE CatID = 6 -- Chocolate lab, change for some other
UNION ALL
SELECT * FROM Category AS a
INNER JOIN Tree t ON a.CatID = t.ParentID
)
SELECT * FROM Tree WHERE CatLevel = 2
Как я могу получить идентификатор узла CatLevel = 2 для всех узлов / листьев?
Спасибо!
1 ответ
Я знаю, что пост довольно старый, но вот как я это сделал.
В моей таблице есть несколько корневых категорий, поэтому я использую @RootId, чтобы указать, какая из них мне нужна. Для корневой категории ParentID установлено значение NULL.
Если в вашей таблице категорий нет столбца CatLevel - cte Tree создаст его для вас. Если вы собираетесь часто использовать запрос, лучше добавить его в таблицу (код UPDATE комментируется в конце).
Если вы хотите иметь динамическое количество столбцов (не фиксированное до 5, как в моем случае), сделайте динамический SQL-запрос для него.
DECLARE @RootId INT, @SearchId INT
SET @RootId = 0
SET @SearchId = 0
--SELECT it.Id, it.ParentId, it.Name, it.RootId FROM ItemTree it
--WHERE (@RootId = 0 OR it.RootId = @RootId) AND (@SearchId = 0 OR it.Id = @SearchId)
;WITH Tree AS --gets category levels
(
SELECT 1 as CatLevel, it.Id, it.ParentId, it.Name, it.RootId
FROM ItemTree it WHERE it.ParentId IS NULL AND (@RootId = 0 OR it.RootId = @RootId) AND (@SearchId = 0 OR it.Id = @SearchId)
UNION ALL
SELECT t.CatLevel + 1 as CatLevel, it.Id, it.ParentId, it.Name, it.RootId
FROM ItemTree it
INNER JOIN Tree t ON it.ParentId = t.Id
WHERE (@RootId = 0 OR it.RootId = @RootId) AND (@SearchId = 0 OR it.Id = @SearchId)
----if CatLevel already exists in the table, just use it instead for better performance
--SELECT it.CatLevel, it.Id, it.ParentId, it.Name FROM ItemTree it
--WHERE (@RootId = 0 OR it.RootId = @RootId) AND (@SearchId = 0 OR it.Id = @SearchId)
)
, LeafIds AS --use it to show only leaf categories
(
SELECT DISTINCT Id FROM ItemTree it WHERE (@RootId = 0 OR it.RootId = @RootId)
EXCEPT
SELECT DISTINCT ParentId FROM ItemTree it WHERE (@RootId = 0 OR it.RootId = @RootId)
)
, LeafTree AS --prepare data for pivots
(
SELECT t.CatLevel, t.Id, t.ParentId, t.Name, Id AS LeafId
FROM Tree t --WHERE Id IN (SELECT Id FROM LeafIds)
UNION ALL
SELECT t.CatLevel, t.Id, t.ParentId, t.Name, lt.LeafId
FROM Tree t
INNER JOIN LeafTree lt ON lt.ParentId = t.Id
)
, TreeFinal AS --make pivots for Id and Name over same leafIds
(
SELECT t.*, pt1.[1] AS ID_1, pt1.[2] AS ID_2, pt1.[3] AS ID_3, pt1.[4] AS ID_4, pt1.[5] AS ID_5
, pt2.[1] AS Name_1, pt2.[2] AS Name_2, pt2.[3] AS Name_3, pt2.[4] AS Name_4, pt2.[5] AS Name_5
, ISNULL(pt2.[1], '') + ISNULL(' - ' + pt2.[2], '') + ISNULL(' - ' + pt2.[3], '') + ISNULL(' - ' + pt2.[4], '') + ISNULL(' - ' + pt2.[5], '') AS Path
FROM Tree t
JOIN (
SELECT LeafId, CatLevel, Id
FROM LeafTree
) source
PIVOT ( MAX(Id) FOR CatLevel IN ([1], [2], [3], [4], [5]) ) AS pt1 ON pt1.LeafId = t.Id
JOIN (
SELECT LeafId, CatLevel, Name
FROM LeafTree
) source
PIVOT ( MAX(Name) FOR CatLevel IN ([1], [2], [3], [4], [5]) ) AS pt2 ON pt2.LeafId = t.Id
)
--Use following to update CatLevel in your table, if it was not there before
--UPDATE it SET CatLevel = Tree.CatLevel FROM Tree JOIN dbo.ItemTree it ON it.Id = Tree.Id
SELECT * FROM TreeFinal ORDER BY Path