在BBS的编写中,经常有人问怎样实现树状结构?一个比较不负责任的回答是:使用递归算法。当然,递归是一个可行的办法

(二叉树的历遍也好象只能使用递归算法),但对于BBS来说,这样做势必要进行大量的Sql查询(虽然可以使用存储过程来做,但要从根本上加快速度,则应该考虑更快的算法)。
下面给出一个可行的彻底摒弃递的实现树状结构的算法。
    下面给出另一种使用“使用中值排序基数法”实现树状结构:
一、主要思想:增加一个排序基数字段ordernum,回复同一根贴的贴子中插入贴子时,排序基数ordernum取两者的中值。
    为了叙述的简洁,在此只讨论与树状结构有关的字段。
在表中增加三个冗余字段,rootid——用于记录根id,deep——用于记录回复的深度(为0时表示根贴),ordernum——排序基数(关键所在)。
表forum与(只列与树状结构有关的字段):id   rootid   deep    ordernum其中id、rootid、deep均为int型(deep可为tinyint型),ordernum为float型。
例:(在此为了简单,使用一个小的起始排序基数,在实际应用中,应使用较大的起始基数,且应取2的整数次幂,如65536=2^16,下面所说的排序均指按ordernum从小到大排序)。
id   rootid    deep    ordernum
1      0        0            0
2      1        1           64
______________________________
3      1        1           32    回复第1贴,取1、2基数的中值即(0+64)/2
排序后结果为:
id   rootid    deep    ordernum
1      0        0            0
3      1        1           32
2      1        1           64
______________________________
4      1        2           48   回复第3贴,取3、2的基数中值即(32+64)/2
排序后结果为:
id   rootid    deep    ordernum
1      0        0            0
3      1        1           32
4      1        2           48
2      1        1           64
______________________________
5      1        3           56 回复第4贴,取4、2的基数中值即(48+64)/2
排序后的结果为:
id   rootid    deep    ordernum
1      0        0            0
3      1        1           32
4      1        2           48
5      1        3           56
2      1        1           64
______________________________
6      1        2           40 回复第3贴,取3、4的基数中值即(32+48)/2
排序后的结果为:
id   rootid    deep    ordernum
1      0        0            0
3      1        1           32
6      1        2           40
4      1        2           48
5      1        3           56
2      1        1           64
这样排序基数ordernum与回复深度deep一起就实现了如下的树状结构:
id
1
 3
     6
     4
       5
2
二、插入的实现(如何确定排序基数,下面所指贴子均为同一根下的子贴)
(一)根ordernum定为0
(二)第一条回复贴子基数定为2的整数次幂(如65536=2^16,可取更大的数)
(三)回复最后一条贴子时,基数取最后一贴的基数ordernum再加上2的整数次幂(同上)
(四)回复中间的贴子时,基数ordernum取前后贴子的基数中值
三、删除的实现
    删除贴子(剪枝)时,只需找出下一个回复深度deep小于或等于要删贴子的回复深度(deep)的贴子,然后将基数ordernum位于两个贴子基数之间的贴子删除即可实现剪枝。
    如上例子中,要删除3贴(基数为32)下的子枝,由于3的深度为1,下一个深度小于或等于1的贴子为2贴(它的基数为64),则只需删除基数在32至64间(64除外)的贴子就行了。也就是删除了3、6、4、5贴。要删其它亦然。
四、显示的实现
    只需执行select * from forum order by rootid+id-sign(rootid)*id desc,ordernum,然后结合deep就可实现树状的显示。
五、具体实现方法(以存储过程为例)
加贴存储过程:(省略注册用户检测以及积分部分内容)
CREATE PROCEDURE [add] @keyid int,@message varchar(50) OUTPUT ———keyid为回复的贴子id号,如果是新贴则为0,@message为出错信息
AS
 IF (@keyid=0)
    INSERT INTO forum (rootid,deep,ordernum,……) values(0,0,0,……)
 ELSE
    BEGIN
     DECLARE @rootid int,@id int,@deep int,@begnum float,@endnum float,@ordernum float
     SELECT @rootid=0,@id=0,@deep=0,@begnum=0,@endnum=0,@ordernum=0
     SELECT @rootid=rootid,@id=id,@begnum=ordernum,@deep=deep from forum where id=@keyid
     IF (@id=0)
       BEGIN
        SELECT @message='要回复的帖子已经被删除!'
        return
       END
     ELSE
       BEGIN
        IF (@rootid=0) SELECT @rootid=@id ——回复的是根贴,取其id为新加贴的rootid
        SELECT @endnum=ordernum where rootid=@rootid and ordernum>@begnum order by ordernum
        IF (@endnum=0)
          SELECT @ordernum=@begnum+65536   ——回复的是最后一贴
        ELSE
          SELECT @ordernum=(@begnum+@endnum)/2 ——关键,取排序基数中值
        INSERT into forum (rootid,deep,ordernum,……) values(@rootid,@deep+1,@ordernum,……)
       END
    END
 Select @message='成功'
 return
剪枝存储过程:(省略注册用户检测以及积分部分内容)
CREATE PROCEDURE [del] @keyid int,@message varchar(50) OUTPUT ———keyid为要删除的贴子id号,如果是新贴则为0,@message为出错信息
AS
DECLARE @rootid int,@id int,@deep int,@begnum float,@endnum float
SELECT @rootid=0,@deep=0,@begnum=0,@endnum=0,@id=0
SELECT @id=id,@begnum=ordernum,@rootid=rootid,@deep=deep from forum where id=@keyid
IF (@id=0)
 BEGIN
   SELECT @message='该帖子不存在!"
   return
 END
ELSE
 BEGIN
   SELECT @endnum=ordernum from forum where rootid=@rootid and deep<=@deep and ordernum>@begnum order by ordernum
   IF (@endnum=0)    ——要删除的是最后一个子枝
     DELETE FROM forum where ordernum>=@begnum and (rootid=@rootid or id=@rootid)
   ELSE
     DELETE FROM forum where ordernum>=@begnum and ordernum<@endnum and (rootid=@rootid or id=@rootid)
 END
显示存储过程(略)
总结:由于省去了childnum字段,因此如果想要知道根贴(或子贴)有多少个子贴,则需使用统计方法或增加对应的字段记录,该问题可不列为树状结构讨论之列。

用中值排序基数法实现树状结构 (转)相关推荐

  1. mysql如何实现树状结构_实现树状结构的两种方法

    实现树状结构的两种方法 实现树状结构的两种方法 1.递归法 递归是指在函数中显式的调用它自身. 利用递归法实现树状结构的特点是写入数据速度较快,显示速度较慢(在树的分支/层次较多的情况下尤其明显).适 ...

  2. oracle 树状结构一直出现不了_SEO站内优化:网站结构优化(十一)

    上文已经对搜索引擎排名机制,以及影响关键词排名的因素进行了全面的讲解,详情查阅SEO入门到精通(十):搜索引擎的排名机制.本文主要讲解站内优化中最为重要的"网站结构优化",为什么说 ...

  3. Java递归子集算法(树状结构)的逻辑和实例代码实现 @杨章隐

    Java递归算法(树状结构)的逻辑和实例 1.应用场景: 递归算法作为一个经常使用的算法,无论在API开发还是计算文件夹都是比较常用的, 在api开发过程中我们经常遇到需要返回树状结构的json 例如 ...

  4. Java处理数据成为树状结构

    如题所示,项目中需要将部分数据处理成为树状结构,实现过程如下: 注:也可以使用sql达到该目的,但此处数据不多,故在代码中处理,主要是sql处理不是很会 // 获取需要封装的数据List<Dat ...

  5. 系统管理模块_部门管理_设计(映射)本模块中的所有实体并总结设计实体的技巧_懒加载异常问题_树状结构...

    系统管理模块_部门管理_设计本模块中的所有实体并总结设计实体的技巧 设计实体流程 1,有几个实体? 一般是一组增删改查对应一个实体. 2,实体之间有什么关系? 一般是页面引用了其他的实体时,就表示与这 ...

  6. oracle树状排序,Oracle树状结构查询

    oracle用表的形式组织数据,某些数据还呈现树状结构,提供了对这些数据的组织.查询等功能.在扫描树结构表时,要依次访问树中的每一个节点,并且每个节点只能访问一次,其步骤如下: 1:从根节点开始 2: ...

  7. SQL SERVER树状结构排序

    1 1.1 1.2 2 2.1 2.1.1 2.1.2 2.2 2.2.1 假如我们查询到树状结构数据,并要求对其排序,排序效果如上图.实现的思路如下:对每一层的节点按照排序字段排序,同一层节点按照排 ...

  8. 前端获取后端的数组对象处理(去重,排序,转为树状结构)

    后端接口数据处理,对数组对象进行 去重(通过元素id去重),排序,转为树状结构 下面为后端转过来的一个扁平数据 var data = [{id: '1',staffName: '陈三三',truePr ...

  9. element表格多列排序_vue表格树状结构的实现

    这绝对是我最后一次写树状结构,我非常的确定一定以及肯定! 上一篇文章总结了vue树状目录组件,忘记的可以点击传送门回顾一下: vue目录树组件 还是这个目录树,改版成 表格树状结构 . 我这里采用的U ...

最新文章

  1. 《Microsoft Sql server 2008 Internals》读书笔记--第九章Plan Caching and Recompilation(10)
  2. cacti0.8.8安装文档
  3. HDU 1325 Is It A Tree? 并查集
  4. VTK:Filtering之TriangulateTerrainMap
  5. java arcgis server_ArcGIS Server Java 开发实战---自定义command
  6. js html转为实体,字符串js编码转换成实体html编码的方法(防范XSS攻击)
  7. 数据仓库ETL之DataX(一)简介
  8. 拥有自我意识、自我模拟的粒子机器人来了
  9. 免费资源下载:暗色色系的超棒搜索框和下拉菜单UI欣赏
  10. 转载:OpenStack从入门到放弃
  11. SQL中 UNION 和 UNION ALL 操作符小结
  12. 安卓开发_使用AlertDialog实现对话框
  13. 惠普打印机墨盒更换教程_惠普打印机安装步骤 打印机墨盒加墨方法
  14. STC 51单片机仿真总结
  15. JavaScript (05)-大话JS,屌丝逆袭(四)
  16. 蒟蒻C语言入门篇(一)
  17. [NOI2010] 航空管制
  18. 行列式用计算机怎么算,行列式计算机方法
  19. Win10 系统设置共享文件
  20. 石墨笔记,为知笔记和Effie哪个更适合学生?

热门文章

  1. 像“打游戏”一样用Numpy,试试?
  2. 退休失败,64岁Python之父决定加入微软,将开源进行到底!
  3. 人民日报:中国 31 个省市区最好的大学
  4. 干货 | 你的Paper阅读能力合格了吗(硕士生版)
  5. 一群猥琐至极的大学舍友,我有点想你们了...附骚图
  6. 程序员最常说的9句话,精准!
  7. 记录一些user-agent
  8. OpenCV中图像修复技术介绍与演示
  9. 模型优化的风向标:偏差与方差
  10. winform改变控件的外形