【发布时间】:2010-10-14 03:44:38
【问题描述】:
我最近一直在使用Nested Set Model 中的废话。我喜欢为几乎所有有用的操作和视图设计查询。我坚持的一件事是如何选择节点的直接子节点(仅子节点,而不是进一步的后代!)。
老实说,我确实知道一种方法 - 但它涉及无法管理的 SQL 数量。我确信有一个更直接的解决方案。
【问题讨论】:
标签: sql tree nested-set-model
我最近一直在使用Nested Set Model 中的废话。我喜欢为几乎所有有用的操作和视图设计查询。我坚持的一件事是如何选择节点的直接子节点(仅子节点,而不是进一步的后代!)。
老实说,我确实知道一种方法 - 但它涉及无法管理的 SQL 数量。我确信有一个更直接的解决方案。
【问题讨论】:
标签: sql tree nested-set-model
您是否阅读了您发布的文章?它在“查找节点的直接下属”标题下
SELECT node.name, (COUNT(parent.name) - (sub_tree.depth + 1)) AS depth
FROM nested_category AS node,
nested_category AS parent,
nested_category AS sub_parent,
(
SELECT node.name, (COUNT(parent.name) - 1) AS depth
FROM nested_category AS node,
nested_category AS parent
WHERE node.lft BETWEEN parent.lft AND parent.rgt
AND node.name = 'PORTABLE ELECTRONICS'
GROUP BY node.name
ORDER BY node.lft
)AS sub_tree
WHERE node.lft BETWEEN parent.lft AND parent.rgt
AND node.lft BETWEEN sub_parent.lft AND sub_parent.rgt
AND sub_parent.name = sub_tree.name
GROUP BY node.name
HAVING depth <= 1
ORDER BY node.lft;
但是,我所做的(这是作弊)是将嵌套集与邻接列表结合在一起——我在表中嵌入了一个“parent_id”,这样我就可以轻松地询问一个节点的子节点。
【讨论】:
在我看来,如果没有子查询或父列冗余,这应该很容易实现!例如,给定父母的左右是已知的:
SELECT child.id
FROM nodes AS child
LEFT JOIN nodes AS ancestor ON
ancestor.left BETWEEN @parentleft+1 AND @parentright-1 AND
child.left BETWEEN ancestor.left+1 AND ancestor.right-1
WHERE
child.left BETWEEN @parentleft+1 AND @parentright-1 AND
ancestor.id IS NULL
也就是说,“从所讨论节点的所有后代中,选择它们与节点之间没有祖先的那些”。
【讨论】:
这个更好更小
用户“bobince”几乎拥有它。我想通了并让它为我工作,因为我比大多数人有更多的 MySQL 经验。但是,我明白为什么 bobince 的回答可能会吓跑人们。他的询问不完整。您需要先将 parent_left 和 parent_right 选择到 mysql 变量中。
下面的两个查询假设您的表名为tree,您的左列名为lft,右列名为rgt,并且您的主键名为id。更改这些值以满足您的需要。此外,检查第一个 select 语句。您会看到我正在查找节点 5 的直接后代。更改数字 5 以查找您想要的任何节点的子节点。
我个人认为这是一个比目前提出的其他查询更时尚、更性感、更有效的查询。
SELECT `lft`, `rgt` INTO @parent_left, @parent_right FROM efm_files WHERE `id` = 5;
SELECT `child`.`id`
FROM `tree` AS `child`
LEFT JOIN `tree` AS `ancestor` ON
`ancestor`.`lft` BETWEEN @parent_left+1 AND @parent_right-1 AND
`child`.`lft` BETWEEN `ancestor`.`lft`+1 AND `ancestor`.`rgt`-1
WHERE
`child`.`lft` BETWEEN @parent_left+1 AND @parent_right-1 AND
`ancestor`.`id` IS NULL
【讨论】:
我知道我在做一个死灵帖子, 但这是我的看法。
为什么不在嵌套集中包含“深度”列? 深度列将指示项目的“级别”。
所以,要选择项目的直接子项,只需这样做
select c.*
from tree as p
join tree as c on (c.left > p.left and c.right < p.right and c.depth = p.dept + 1)
where p.id = @parentID
【讨论】:
我也会选择深度列。但是使用
SELECT Child.Node, Child.LEFT, Child.RIGHT
FROM Tree AS Child, Tree AS Parent
WHERE
Child.Depth = Parent.Depth + 1
AND Child.LEFT > Parent.LEFT
AND Child.RIGHT < Parent.RIGHT
AND Parent.LEFT = 1 -- Given Parent Node Left Index
【讨论】:
我发现Wikipedia link 有很好的最小化答案版本以及选定的答案。
SELECT DISTINCT Child.Name
FROM ModelTable AS Child, ModelTable AS Parent
WHERE Parent.Lft < Child.Lft AND Parent.Rgt > Child.Rgt -- associate Child Nodes with ancestors
GROUP BY Child.Name
HAVING MAX(Parent.Lft) = @parentId -- Subset for those with the given Parent Node as the nearest ancestor
还有,大家尝试用Linq表达,请点击链接:https://stackoverflow.com/a/25594386/361100
【讨论】: