jiayongmao

使用SQL2005 递归查询结合Row_Number()实现完全SQL端树排序

 

在实际应用中,我们经常需要用到树型结构功能,数据库结构一般如下

即用一个ParentID来标识该节点从属关系。为了最终生成一棵树,一般做法是把记录选出来,然后在程序里递归重新排好序后再呈现出来,但是如果有大量数据,就带来了性能开销问题。那么能不能直接在数据库利用SQL语句排好树后再输出呢?

SQL2005 有个递归查询功能也就是WITH .. AS 语句。对上面这样的表格使用递归查询,可以查询得到某树支下(包括根)的所有节点记录。类似语句如下:

with RelClass 
as 

select 
* from CMS_Site_Class where ClassID = 1
union all 
select csc.
* from CMS_Site_Class as csc inner join RelClass as rc on csc.ClassID_Parent = rc.ClassID ) 
SELECT 
* from RelClass

将得到ClassID为1的根节点下的所有记录:

但是这个记录集显然没有经过树排序,这时还需要程序里进一步处理才能输出到客户端。在这里我介绍一种WITH 结合 Row_Number() 实现SQL端排序的方法。

先来看看最终的代码:

Code

 

-- =============================================
-- Author:    <kingimg>
-- Create date: <2009-2-5>
-- Description:    <生成已排序的树>
-- =============================================
Create PROCEDURE [dbo].[pCMS_Site_Class__GetList] 
@ClassID 
int 
AS 
BEGIN 
with RelClass 
as 

select 
*,0 as Level,cast('0' as nvarchar(max)) as treepath from CMS_Site_Class where ClassID = @ClassID 
union all 
select csc.
*,rc.[Level] + 1,rc.treepath + dbo.Lpad(Row_Number() over (order by csc.OrderID desc),8as treepath from CMS_Site_Class as csc inner join RelClass as rc on csc.ClassID_Parent = rc.ClassID ) 
SELECT 
* from RelClass order by treepath 
END 

 

执行以上存储过程,最后就输出结果:

这棵树已经从上到下按树结构排好序了!程序里只要原样输出即可!

posted on 2009-02-05 17:40  贾永茂  阅读(397)  评论(0编辑  收藏  举报

导航