我正在使用嵌套集(也称为修改的预订树遍历)来存储组列表,我正在尝试找到一种快速方法来为所有组生成面包屑(作为字符串,而不是表)立刻。我的数据也使用邻接列表模型存储(有触发器使两者保持同步)。
例如:
ID Name ParentId Left Right
0 Node A 0 1 12
1 Node B 0 2 5
2 Node C 1 3 4
3 Node D 0 6 11
4 Node E 3 7 8
5 Node F 4 9 9
代表树:
我希望能够有一个返回表的用户定义函数:
ID Breadcrumb
0 Node A
1 Node A > Node B
2 Node A > Node B > Node C
3 Node A > Node D
4 Node A > Node D > Node E
5 Node A > Node D > Node F
为了使这稍微复杂一些(虽然它有点超出了问题的范围),我也有需要遵守的用户限制。因此,例如,如果我只能访问id = 3,那么当我运行查询时,我应该得到:
ID Breadcrumb
3 Node D
4 Node D > Node E
5 Node D > Node F
我有一个用户定义的函数,它将userid作为参数,并返回一个包含所有有效组的ID的表,只要在查询中的某个地方
WHERE group.id IN (SELECT id FROM dbo.getUserGroups(@userid))
它会起作用。
我有一个可以执行此操作的现有标量函数,但它不适用于任何合理数量的组(在2000个组上花费> 10秒)。它将groupid和userid作为参数,并返回nvarchar。它找到给定的组父项(1个查询用于获取左/右值,另一个用于查找父项),将列表限制为用户有权访问的组(使用与上面相同的WHERE子句,因此还有另一个查询),然后使用游标遍历每个组并将其附加到字符串,最后返回该值。
我需要一种能够快速运行的方法(例如,< = 1s)。
这是在SQL Server 2005上。
答案 0 :(得分:3)
这是我从树中的任何一点获取“breadcrumb”路径的SQL。希望它有所帮助。
SELECT ancestor.id, ancestor.title, ancestor.alias
FROM `categories` child, `categories` ancestor
WHERE child.lft >= ancestor.lft AND child.lft <= ancestor.rgt
AND child.id = MY_CURRENT_ID
ORDER BY ancestor.lft
凯丝
答案 1 :(得分:3)
确定。这适用于MySQL,而不是SQL Server 2005.它使用带有子查询的GROUP_CONCAT。
这应该将完整的面包屑作为单列返回。
SELECT
(SELECT GROUP_CONCAT(parent.name SEPARATOR ' > ')
FROM category parent
WHERE node.Left >= parent.Left
AND node.Right <= parent.Right
ORDER BY Left
) as breadcrumb
FROM category node
ORDER BY Left
答案 2 :(得分:2)
如果可以,请使用路径(或者我认为我听说它称为血统)字段:
ID Name ParentId Left Right Path
0 Node A 0 1 12 0,
1 Node B 0 2 5 0,1,
2 Node C 1 3 4 0,1,2,
3 Node D 0 6 11 0,3,
4 Node E 3 7 8 0,3,4,
5 Node F 4 9 9 0,3,4,
要获得节点D及以后(伪代码):
path = SELECT Path FROM Nodes WHERE ID = 3
SELECT * FROM Nodes WHERE Path LIKE = path + '%'
答案 3 :(得分:1)
我最终做的是创建一个大型连接,简单地将此表与自身联系起来,为每个级别反复进行。
首先,我只使用第一级组填充表@topLevelGroups(如果只有一个根可以跳过此步骤),然后使用用户可以看到的组填充@userGroups。
SELECT groupid,
(level1
+ CASE WHEN level2 IS NOT NULL THEN ' > ' + level2 ELSE '' END
+ CASE WHEN level3 IS NOT NULL THEN ' > ' + level3 ELSE '' END
)as [breadcrumb]
FROM (
SELECT g3.*
,g1.name as level1
,g2.name as level2
,g3.name as level3
FROM @topLevelGroups g1
INNER JOIN @userGroups g2 ON g2.parentid = g1.groupid and g2.groupid <> g1.groupid
INNER JOIN @userGroups g3 ON g3.parentid = g2.groupid
UNION
SELECT g2.*
,g1.name as level1
,g2.name as level2
,NULL as level3
FROM @topLevelGroups g1
INNER JOIN @userGroups g2 ON g2.parentid = g1.groupid and g2.groupid <> g1.groupid
UNION
SELECT g1.*
,g1.name as level1
,NULL as level2
,NULL as level3
FROM @topLevelGroups g1
) a
ORDER BY [breadcrumb]
这是一个相当大的黑客,显然限于一定数量的级别(对于我的应用程序,我可以选择一个合理的限制),问题是支持的级别越多,它就增加了以指数方式连接,因此速度慢得多。
在代码中执行它肯定更容易,但对我来说这并不总是一个选项 - 有时候我需要直接从SQL查询中获取它。
我接受这个作为答案,因为这是我最终做的事情,它可能适用于其他人 - 但是,如果有人能提出更有效的方法,我会改变它们。
答案 4 :(得分:1)
我修改了凯西的声明以获得每个元素的面包屑
SELECT
GROUP_CONCAT(
ancestor.name
ORDER BY ancestor.lft ASC
SEPARATOR ' > '
),
child.*
FROM `categories` child
JOIN `categories` ancestor
ON child.lft >= ancestor.lft
AND child.lft <= ancestor.rgt
GROUP BY child.lft
ORDER BY child.lft
随意添加WHERE条件,例如
WHERE ancestor.lft BETWEEN 6 AND 11
答案 5 :(得分:0)
没有特定于sql server的代码,但你只是在寻找:
SELECT * FROM table WHERE left&lt; (currentid.left)和右&gt; (currentid.right)