一道 SQL 题 ... (关于树型结构的在关系表中的存储及其应用处理)

树型结构

相关讨论连接:
http://expert.csdn.net/Expert/TopicView1.asp?id=1477009
原题:
表:
Tree (ID [Integer],ParentID [Integer],Remark [varchar])

INSERT INTO Tree (ID,ParentID)
       SELECT 1,0
    UNION ALL
       SELECT 2,1
    UNION ALL
       SELECT 3,1
    UNION ALL
       SELECT 4,2
    UNION ALL
       SELECT 5,4
    UNION ALL
       SELECT 6,5
    UNION ALL
       SELECT 7,2

T(F1,......)
 INSERT INTO T (F1)
       SELECT 1
    UNION ALL
       SELECT 5
    UNION ALL
       SELECT 3
    UNION ALL
       SELECT 4
    UNION ALL
       SELECT 1
    UNION ALL
       SELECT 7
    UNION ALL
       SELECT 6
    UNION ALL
       SELECT 4
    UNION ALL
       SELECT 5
    UNION ALL
       SELECT 3
    UNION ALL
       SELECT 4
    UNION ALL
       SELECT 1
    UNION ALL
       SELECT 7
    UNION ALL
       SELECT 6
    UNION ALL
       SELECT 4

参考 Tree 表中的父子关系,"祖先"的记录数要包括所有"后代"的记录数,统计 T 表中 F1 各个取值的记录数?
ID      Counts
1       15
2       10
3       2
4       8
5       4
6       2
7       2

答案及简单分析:

/*
看了前几个人的答案,似乎都把问题想复杂了"游标"、"临时表"、"递归"。
"游标"、"临时表" 完全可以不用!
"递归" 思想当然应是解决树型结构的该想到的方法!
但是 T-SQL 的嵌套层次最多只能到 32!
icevi(按钮工厂) 的建议是非常值得提倡的,尽管 ID,ParentID 对于仅存储是足够经济的,
但是若用其提供表现形式,性能的确不会太好!
许多高效的树型结构论坛也确实是存储并维护各个节点的层次信息的数据,这样
显示起来仅需一条 SQL 即可!
下面是我的参考答案,两个自定义函数功能几乎一样,都是运算出前面所提的,
应最好主动维护的"层次信息":

方法一: UDF 递归实现! 有 32 层嵌套限制
*/

alter  FUNCTION dbo.Get32Ancestors
(@X integer)
RETURNS VARCHAR(250)
AS
BEGIN
DECLARE @ID integer
DECLARE @ReturnValue VARCHAR(250)

SELECT TOP 1 @ID = ParentID
FROM tree
WHERE [id] = @X

IF @ID <> @X
   BEGIN
     SELECT @ReturnValue  = cast(ISNULL(dbo.Get32Ancestors(@ID),'') as varchar) + '-'+ cast(@X as varchar)
   END
ELSE SET @ReturnValue = @ID

RETURN @ReturnValue
END

go
/*
2003-3-5
方法二: 无任何限制,若层次太深,效率当然不会高(好像也没更好的办法)
改进了一下:
1.正常节点均从0显示! 0-1-3

2.断码 显示 -7-8-9-10
3.GetAllAncestors(不存在的节点)返回NULL
4.GetAllAncestors(根节点)返回 0-自己
5.死循环点显示: 4-5-6-4-8

*/

alter function GetAllAncestors (@X integer)
returns varchar(1000)
as
begin
declare @ReturnValue  varchar(1000)
declare @ID integer
declare @ParentID integer

set @ID = -1

select top 1 @ID=isnull([ID],0),@ParentID = isnull([ParentID],0)
from tree
where ID = @X

while @id <> @parentid and @parentid <> 0 and @ID >0
      and '-' + isnull(@ReturnValue,'') +'-' not like '%-' + cast(@id as varchar) + '-%'
  begin
    if  @ReturnValue is not null
        set @ReturnValue = '-' + @ReturnValue
    set @ReturnValue= cast(@id as varchar) + isnull(@ReturnValue,'')
    set @id = -1
    select top 1 @ID=isnull([ID],0),@ParentID = isnull([ParentID],0)
      from tree
     where ID = @parentid
  end

set @ReturnValue = '-' + @ReturnValue

if @id>0
   set @ReturnValue = cast(@id as varchar) + isnull(@ReturnValue,'')

if @parentid =0 or  @id = @parentid
   set @ReturnValue = '0-'  + isnull(@ReturnValue,'')  

return(@ReturnValue)
--select dbo.GetAllAncestors(10)
end

go

/*
方法一是"高手"的惯性思维把简单的问题搞复杂了,"太累"!
方法二是思路简单清晰,不但是"菜鸟"首选,"高手"也应反思!

若是本题分为两问:
1.求各节点层次信息
2.求属各节点含后代的记录数

可能大家就会受到一些启发!
函数定义完,下面就应该和 icevi(按钮工厂) 同志的答案异曲同工、不谋而和了
*/

select id,dbo.GetAllAncestors(id)
       ,(select count(*)
           from T
          where '-' + dbo.GetAllAncestors(f1) + '-' like '%-' + cast(tree.id as varchar) + '-%')
from tree

select id,dbo.Get32Ancestors(id)
       ,(select count(*)
           from T
          where '-' + dbo.Get32Ancestors(f1) + '-' like '%-' + cast(tree.id as varchar) + '-%')
from tree

/*
另外还要说一下封装的程度的问题,具体情况具体分析,
本题就不适合定义函数直接得到最终结果!
以上答案仅供参考!!
欢迎继续参与讨论!
*/

时间: 2024-08-04 13:20:42

一道 SQL 题 ... (关于树型结构的在关系表中的存储及其应用处理)的相关文章

AJAX实现动态树型结构

ajax|动态|树型结构 树型结构是一类应用非常广泛的数据结构.人类社会中宗族的族谱和现代企业的组织形式都是树型结构.在计算机领域中,文件系统中文件的管理结构.存储器管理中的页表.数据库中的索引等也都是树型结构.随着Internet的飞速发展,树型结构在浏览器/服务器(Browser/Server,简称B/S)应用系统的应用也越来越广泛. 目前,在互联网上广泛存在.应用的树型结构一般分为两种:静态和动态结构.静态结构存在最多.实现简单,但是静态导致不能改变树的结构和内容,无法反映树的节点信息的变

基于AJAX的动态树型结构的设计与实现

ajax|动态|设计|树型结构 <B>摘 要</B>:简要介绍了一种通用的,动态树型结构的实现方案,该方案基于Asynchronous JavaScript and XML,结合Struts框架设计实现了结构清晰.扩展性良好的多层架构,数据存储于数据库,结合XML描述树的节点信息,使得任何按预定的XML文档描述的信息都可以通过动态树来展现.<br /><table border="0" cellspacing="0" cel

树型结构在ASP中的简单解决

解决|树型结构 树型结构在我们应用程序中还是很常见的,比如文件目录,BBS,权限设置,部门设置等.这些数 据信息都采用层次型结构,而在我们现在的关系型数据库中很难清淅表达.那么要在程序中遇到树型 结构问题该如何处理呢? 最近笔者通过一个ASP权限管理的程序轻松解决了一这问题,现在将其整理出来以飨读者. 首先,要将层次型数据模型转化为关系型数据模型.也就是说如何在我们的ACCESS,SQL SERVER ,ORACLE等关系型数据库中设计这个数据结构. 拿个实例来讲吧,譬如下面一个数据: 文档管理

实现树型结构(一)

树型结构 实现树型结构(第一部分)作者:ACE      最后更新:06/08/2000      类别:原创 先看一下示例,如果你感觉尚可,就继续阅读本文http://www.coolbel.com/ace/articles/test/msdn.asp. 1. 简述 对于大家来说树型结构是很熟悉的一种模型.它的应用十分广泛,比如组织结构,物料清单,资料档案管理,资产管理等等都是以树型结构为基础.在现实生活中,有许多事物可以抽象为树状结构.这种结构可以简化对某些事物的理解,使概念清晰. 2. 表

非递归法实现论坛树型结构及分页!!(心血结晶啊,呵呵)

递归|分页|树型结构 现将本人的实践结果show给大家,不足之处就是分页的方法不太好,不能显示具体的页数,可实在又没有其它更好的解决办法,只好先如此了,如果哪位有类似本论坛的分页方法,表赐教一二,二泉不胜感激!具体可访问我的个人小网站:http://web.nyist.net/~wbgwrq,不废话了,开始吧...... //表的结构如下://creat.sql//简单说明:RootId 论题序数;Layer:帖子层次,缩进的依据:Orders:帖子的顺序CREATE TABLE over_po

非递归法实现论坛树型结构及分页

递归|分页|树型结构 现将本人的实践结果show给大家,不足之处就是分页的方法不太好,不能显示具体的页数,可实在又没有其它更好的解决办法,只好先如此了,如果哪位有类似本论坛的分页方法,表赐教一二,二泉不胜感激!具体可访问我的个人小网站:http://web.nyist.net/~wbgwrq,不废话了,开始吧...... //表的结构如下://creat.sql//简单说明:rootid 论题序数;layer:帖子层次,缩进的依据:orders:帖子的顺序create table over_po

不用递归实现论坛树型结构的算法

递归|树型结构|算法 <jsp:useBean id="mybbs" scope="session" class="netzero.mydb" /> <%@ page contentType="text/html;charset=gb2312" %> <%@ page import="java.io.*" %> <%@ page import="java.

dtree和jquery构建树型结构

对于小型的树型应用来说,dtree是一个不错的选择. 先看一眼dtree给的例子 构造静态树 首先引入css文件和js文件 <link rel="StyleSheet" href="dtree.css" type="text/css" /> <script type="text/javascript" src="dtree.js"></script> 构造静态树其实很简单

带数据库的ajax+asp无限级分类树型结构,好东西别错过!

跟大家分享一下自己写的一个树型结构,参考了动力文章的无限极分类树形结构数据库,看演示吧http://asptree.guaishi.org/aspajax/ 下面是核心类代码,注释是后来加的,可能有些写的不太正确 复制内容到剪贴板 代码: <%'数据库字段为类属性,添加.删除.修改.操作检查等函数为类的方法Class Cls_Leibie    Private nClassID,sClassName,nParentID,sParentPath,nDepth,nRootID,nChild,nOrd