SQL Server存储层级数据实现无限级分类



由于数据库存储的数据都是以平面方式存储,所以目前大部分论坛和其他程序都是用递归来展现层次数据的,如果分类的层次十分深的话那么使用的递归次数相当可观,对性能的影响也非常大。最近要做一个分类信息的平台就遇到这个问题了,那么如何实现快速的展现分层数据呢?MySQL 的开发者帮我们想到了一个算法,这个算法目前唯一的问题就是尚未实现分类排序,我们可以通过右值的反向排序实现先入先出的排序。在这里我们需要了解的是如何用 SQL Server 来实现,我们就以省市县数据库为例来实现:

  SQL Server存储层级数据实现无限级分类

  如图所示我们将一个树节点的左右各编上号码,就可以看出一些规律,山西的左右值为(8,17),那么所有左值大于8,右值小于17的节点都是属于山西的子节点。稷山先的左右值为(14,15),那么他的所有父节点就是左值小于14,右值大于15的节点,怎么样,用这个方法实现的无限级分类性能绝对是顶呱呱的。一次查询就可以查出属于某个节点的数据以及他子节点的数据。这个算是我见过性能最高的无限级分类算法。其他算法跟这个对比基本没有任何优势。

  我们先建立一个数据表,结构如下图(LID 为左值,RID 为右值,Tree 为节点深度,Name 和 ID 就不多说了,节点的索引和名称)

  SQL Server存储层级数据实现无限级分类

  我们可以使用下面的存储过程来获得一个节点和其子节点:

  • CREATE PROCEDURE CLSP_ZoneSelect  
  • (  
  •     @Root INT,  
  •     @Tree INT 
  • )  
  • AS 
  •     SELECT Z.ID,Z.Tree,Z.Name  
  •     FROM CL_ZoneData AS Z,CL_ZoneData AS P  
  •     WHERE   P.ID = @Root  
  •             AND Z.LID >= P.LID AND Z.RID <= P.RID  
  •             AND (@Tree = 0 OR Z.Tree <= P.Tree + @Tree)  
  •     ORDER BY Z.LID ASC 
  • GO 
     

  我们可以用下面这个存储过程来在一个节点下插入新的子节点:


  • CREATE PROCEDURE CLSP_ZoneInsert  
  • (  
  •     @Root INT,  
  •     @Name NVARCHAR(50)  
  • )  
  • AS 
  •     DECLARE @RID AS INT,@NID AS INT,@Tree AS INT 
  •  
  •     SET @RID = 1  
  •     SET @NID = 0  
  •     SET @Tree = 1  
  •  
  •     IF @Root = 0  
  •     BEGIN 
  •         SELECT TOP 1 @RID = RID + 1  
  •         FROM CL_CateData ORDER BY RID DESC 
  •     END 
  •     ELSE 
  •     BEGIN 
  •         SELECT @RID = RID, @Tree = Tree + 1  
  •         FROM CL_ZoneData WHERE ID = @Root  
  •     END 
  •  
  •     IF @Root = 0 OR @RID > 1  
  •     BEGIN 
  •         UPDATE CL_ZoneData SET RID = RID + 2 WHERE RID >= @RID  
  •         UPDATE CL_ZoneData SET LID = LID + 2 WHERE LID > @RID  
  •  
  •         INSERT INTO CL_ZoneData(LID,RID,Tree,Name)  
  •         VALUES (@RID,@RID + 1,@Tree,@Name)  
  •  
  •         SET @NID = SCOPE_IDENTITY()  
  •     END 
  •     SELECT @NID  
  • GO 

  删除一个节点可以用下面的存储过程:


  • CREATE PROCEDURE CLSP_ZoneDelete  
  • (  
  •     @ID INT 
  • )   
  • AS 
  •     DECLARE @LID AS INT, @RID AS INT, @WID AS INT, @DID AS INT 
  •     SET @DID = 0  
  •     SELECT @DID = ID, @LID = LID, @RID = RID, @WID = RID - LID + 1 FROM CL_ZoneData WHERE ID = @ID  
  •     IF @DID != 0  
  •     BEGIN 
  •         DELETE FROM CL_ZoneData WHERE LID BETWEEN @LID AND @RID  
  •         UPDATE CL_ZoneData SET RID = RID - @WID WHERE RID > @RID  
  •         UPDATE CL_ZoneData SET LID = LID - @WID WHERE LID > @RID  
  •     END 
  •     SELECT @DID  
  • GO  

(资料:来自 http://www.mscto.com/SqlServer/2009090294165.html)

转载于:https://www.cnblogs.com/Jefft/archive/2009/09/03/1559664.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值