查找嵌套集的面包屑

时间:2009-04-30 21:14:56

标签: sql sql-server-2005 breadcrumbs nested-sets mptt

我正在使用嵌套集(也称为修改的预订树遍历)来存储组列表,我正在尝试找到一种快速方法来为所有组生成面包屑(作为字符串,而不是表)立刻。我的数据也使用邻接列表模型存储(有触发器使两者保持同步)。

例如:

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

代表树:

  • 节点A.
    • 节点B.
      • 节点C
    • 节点D.
      • 节点E
      • 节点F

我希望能够有一个返回表的用户定义函数:

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上。

6 个答案:

答案 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)