xuzuning(唠叨):老师,请说详细点可以么,学生不懂呀

解决方案 »

  1.   

    先看看这篇东西,有个印象:
    用中值排序基数法实现树状结构——让递归滚一边去 
     廖家远 发表于 2000-07-26 03:07:25 Joy ASP ←返回版面   用中值排序基数法实现树状结构    在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
      

  2.   

    先看看这篇东西,有个印象:
    用中值排序基数法实现树状结构——让递归滚一边去 
     廖家远 发表于 2000-07-26 03:07:25 Joy ASP ←返回版面   用中值排序基数法实现树状结构    在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
      

  3.   

    剪枝存储过程:(省略注册用户检测以及积分部分内容)
    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字段,因此如果想要知道根贴(或子贴)有多少个子贴,则需使用统计方法或增加对应的字段记录,该问题可不列为树状结构讨论之列。“中值排序基数法实现树状结构”的补充    由于一时疏忽,造成了此法“对于int类型的基数字段,对原始贴的回复只能有31个;numeric类型的基数字段,对原始贴的回复也不能超过120个”(实际上是对于int型字段,原始贴的回复第32个以上的树状结构显示开始紊乱,对于numeric型的基数字段,原始贴的回复从121个以上树状结构显示开始紊乱——回复并不会出问题),这是由于计算机存储精度引起的。
        我们可以将加贴的存储过程修改一下(加进前面加上**号的行)以限制到了一定深度(在特定数据类型下,基数无法分辨)的时候不再以树状结构显示(而采用平显——平行显示,这样做虽然有点象折衷的做法,但在实际上由于浏览器等的限制——即使在深度100的时候能以树状结构显示,但从你的浏览器看来的树状结构的结果仍然不是清晰的——屏幕宽度不够,会折行呗)。加贴存储过程: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   ——回复的是最后一贴,可以在此限制@ordernum的范围以防溢出
            ELSE
    **          BEGIN
    **            IF @endnum-@begnum>1           ——精度仍能分辨。此处的1为精度标记,适合于基数字段为int,如果基数字段为numeric字段,请酌情选娶(呸呸呸,错别字来了),目的是使基数精度过小时限制深度增加,避免显示时的紊乱
    **              SELECT @ordernum=(@begnum+@endnum)/2,@deep=@deep+1  ——关键,取排序基数中值
    **            ELSE
    **              SELECT @ordernum=@begnum     ——限制深度不能再增加,此贴与回复贴平行显示,如果存在parentid字段,则要取parentid和回复贴的parentid一样
    **          END 
    **        INSERT into forum (rootid,deep,ordernum,……) values(@rootid,@deep,@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
    **     BEGIN
    **       IF @begnum=@endnum
    **         DELETE FROM forum where id=@id and (rootid=@rootid or id=@rootid) ——已经受精度限制的枝,只删当前贴
    **       ELSE
    **         DELETE FROM forum where ordernum>=@begnum and ordernum<@endnum and (rootid=@rootid or id=@rootid)
    **     END
      END
       虽然是限制,但此限制应该是必须的,因为实际上的回复深是不能太大的(就象我在这里灌到八九层的时候,讨饭猫就大叫“打住打住”了,呵呵)   欢迎访问我的个人主页http://swuse.yeah.net(原来bigeagle是说我这一句话“目的明显啊”)