使用“使用中值排序基数法”实现树状结构(一)

原创 2005年02月27日 22:39:00

 在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字段,因此如果想要知道根贴(或子贴)有多少个子贴,则需使用统计方法或增加对应的字段记录,该问题可不列为树状结构讨论之列。

版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

Map.Entry 类使用简介,为Map中值排序的方法

使用Map.Entry类,可以同时得到键和值的信息。 先看标准的Map访问方法如下: Set keys = map.keySet( ); if(keys != null) { Ite...

使用邻接表实现基数排序

使用邻接表实现基数排序可以节省内存使用空间,但是在排序上会有问题,解决的办法就是在分配数组的时候从最后一个数组开始分配,收集的时候从第一个开始收集就可以解决了。 实现代码如下: 注:使用不同的编译...

采用Vivado HLS为视频处理实现中值滤波器和排序网络

Vivado的高层次综合功能将帮助您为嵌入式视频应用设计更好的排序网络。 从汽车到安全系统再到手持设备,如今采用嵌入式视频功能的应用越来越多。每一代新产品都需要更多的功能和更好的图像质量。但是,...

快速排序-常见中轴(主元pivot)选择方法及实现代码(末位/随机/三数中值/..)

一、选取最后一个元素 在我们的课本中,看到最多的就是选择第一个元素作为中轴,但是在很多书上却选择最后一个元素作为中轴。下面就让我们来一睹选取最后一个元素作为中轴的快排。 注:本...

快速排序普通法与中值枢纽元法以及三切法

快速排序影响速度的一个因素便是枢纽元,在这里第一种方法为将第一个元素作为枢纽元,这样做有一个弊端,就是当顺序为逆序时或者大量重复的数时T(N) = O(N^2),方法一是中值枢纽法,即将第一个,中间的...

数据结构算法之排序系列Java、C源码实现(8)--基数排序

基数排序    这是一种和前述算法完全不同的排序方法。前述算法都要进行关键字的比较,而基数排序不需要进行记录关键字之间的比较。  链式基数排序   思想:从低位到高位一次对待排序的关键码进行分配和收集...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)