zoukankan      html  css  js  c++  java
  • 数据结构5(树)

    第5章  树

    【例5-1】写出如图5-1所示的树的叶子结点、非终端结点、每个结点的度及树深度。

     
     

    解:

    (1)叶子结点有:B、D、F、G、H、I、J。

    (2)非终端结点有:A、C、E。

    (3)每个结点的度分别是:A的度为4,C的度为2,E的度为3,其余结点的度为0。

    (4)树的深度为3。

    【例5-2】一棵度为2的树与一棵二叉树有什么区别?

    解:度为2的树有两个分支,但分支没有左右之分;一棵二叉树也有两个分支,但有左右之分,左右子树的次序不能交换。

    【例5-3】树与二叉树有什么区别?

    解:区别有两点:

    (1)二叉树的一个结点至多有两个子树,树则不然;

    (2)二叉树的一个结点的子树有左右之分,而树的子树没有次序。

    //二叉树有顺序,最多度为二

    【例5-4】分别画出具有3个结点的树和三个结点的二叉树的所有不同形态。

    解:如图5-2(a)所示,具有3个结点的树有两种不同形态。

     
       

     如图5-2(b)所示,具有3个结点的二叉树有以下五种不同形态。

     
       

    【例5-5】在一棵度为m树中,度为1的结点数为n1,度为2的结点数为n2,……,度为m的结点数为nm,则该数中含有多少个叶子结点?有多少个非终端结点?

    解:设度为0的结点(即叶子结点)数目为n0,树中的分支数为B,树中总的结点数为N,则有:

    (1)从结点的度考虑:

    N= n0+ n1+ n2+……+nm

    (2)从分支数目考虑:一棵树中只有一个根结点,其他的均为孩子结点,而孩子结点可以由分支数得到。所以有:

                             N=B+1=0×n0+1×n1+2×n2+…+m×nm+1。。。。

            由以上两式,可得

                                      n0+ n1+ n2+……+nm=0×n0+1×n1+2×n2+…+m×nm+1

            从而可导出叶子结点的数目为:

                                      n0=0×n1+1×n2+…+(m-1)×nm+1=1+。。。。。。

            从而可以得到非终端结点的数目为

                                      N- n0= n1+ n2+……+nm=。。。。。。。

    【例5-6】一棵含有n个结点的k叉树,可能达到的最大深度和最小深度各为多少?

    解:(1)当k叉树中只有一层的分支数为k,其它层的分支数均为1时,此时的树具有最大的高度,为:n-k+1。//n-k是分支为1,1是分支为k

    (2)当该k叉树为完全k叉树时,其深度最小。参照二叉树的性质4可知,其深度为:

                             +1。

    【例5-7】证明任何一棵满二叉树T中的分支数B满足B=2(n0-1)(其中n0为叶子结点数)。

    证明:

    ∵T为满二叉树

    ∴不存在度为1的结点

    设该二叉树中总的结点数为n,度为2的结点总数为n2,分支数为B

    则有n=n0+ n2                                ①

    又∵除了根结点外,其余n-1个结点都有一个分支进入,即有n个结点的二叉树共有n-1条边

    ∴n=B+1                                           ②

    由①、②两式,可得 B+1=n0+ n2                    ③

    又由二叉树的性质3可知             n2=n0-1          ④

    由③、④两式可知    B= n0+ n0-1-1=2(n0-1) 求证成立。

    //这题真的很想卧槽啊

    【例5-8】如图5-3所示的二叉树,试分别写出它的顺序表示和链接表示(二叉链表)。

    //二叉树的顺序表示和链式表示

    解:

    (1)顺序表示。

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    11

    a

    b

    c

    d

    e

    ^

    ^

    ^

    ^

    f

    g

    (2)该二叉树的二叉链表表示如图5-4所示。

     
       

     

    【例5-9】试找出满足下列条件的所有二叉树:

    (1)先序序列和中序序列相同;

    (2)中序序列和后序序列相同;

    (3)先序序列和后序序列相同。

    解:

    (1)先序序列和中序序列相同的二叉树为:空树或者任一结点均无左孩子的非空二叉树;(右单支树)

    (2)中序序列和后序序列相同的二叉树为:空树或者任一结点均无右孩子的非空二叉树;(左单支树)

    (3)先序序列和后序序列相同的二叉树为:空树或仅有一个结点的二叉树。

    【例5-10】如图5-5所示的二叉树,要求:

    (1)写出按先序、中序、后序遍历得到的结点序列。

    (2)画出该二叉树的后序线索二叉树。

    解:

    (1)      先序遍历序列:ABDEFC

                     中序遍历序列:DEFBAC

    后序遍历序列:FEDBCA

    (2)其后序线索二叉树如图5-6所示。//暂时只考中序线索二叉树

     
       

     

     【例5-11】将图5-7所示的树转换为二叉树。

     

    解:第一步,加线。第二步,抹线。第三步,旋转。过程如图5-8所示。

     
       
     
       


    【例5-12】将如图5-9所示的二叉树转换为树。

    解: 第一步,加线。第二步,抹线。第三步,调整。过程如图5-10所示。

     

    【例5-13】将如图5-11所示的森林转换成二叉树。//左孩子右兄弟

     

    解: 步骤略,结果如图5-12所示。

     

     
       

    【例5-14】假定用于通信的电文由8个字符A、B、C、D、E、F、G、H组成,各字母在电文中出现的概率为5%、25%、4%、7%、9%、12%、30%、8%,试为这8个字母设计哈夫曼编码。

    解: 根据题意,设这8个字母对应的权值分别为(5,25,4,7,9,12,30,8),并且n=8。

    (1)设计哈夫曼树的步骤如图5-13所示。//它一直保持第一层是要排序的数所以不会乱

     
       

     

    (2)设计哈夫曼编码

    利用第八步得到的哈夫曼树,规定左分支用0表示,右分支用1表示,字母A、B、C、D、E、F、G、H的哈夫曼编码如下表示:

    A:0011      B:01            C:0010      D:1010

    E:000       F:100       G:11            H:1011

    习题5

    一、单项选择题

    1. 在一棵度为3的树中,度为3的结点数为2个,度为2的结点数为1个,度为1的结点数为2个,则度为0的结点数为(  C)个。//1+2+3=6

    A. 4         B. 5           C. 6           D. 7

    2. 假设在一棵二叉树中,双分支结点数为15,单分支结点数为30个,则叶子结点数为(B )个。//n0=n2+1

    A. 15        B. 16           C. 17          D. 47

    3. 假定一棵三叉树的结点数为50,则它的最小高度为(C )。//公式算不出来就画一画

    A. 3         B. 4           C. 5           D. 6

    4. 在一棵二叉树上第4层的结点数最多为(  D)。

    A. 2         B. 4           C. 6           D. 8

    5. 用顺序存储的方法将完全二叉树中的所有结点逐层存放在数组中R[1..n],结点R[i]若有左孩子,其左孩子的编号为结点(B)。//左孩子2i右孩子2i+1;若2i/2i+1>n,无左右孩子

    A. R[2i+1]   B. R[2i]       C. R[i/2]      D. R[2i-1]

    6. 由权值分别为3,8,6,2,5的叶子结点生成一棵哈夫曼树,它的带权路径长度为(D  )。

    A. 24        B. 48          C. 72          D. 53

    //在这个树构造这种题上,你已经因为同一原因出错好几次了。原因:两个数字加和之后,要代替两个数字放回原来数列中重新选择。

    7. 线索二叉树是一种( C)结构。

    A. 逻辑       B. 逻辑和存储   C. 物理         D. 线性

    8. 线索二叉树中,结点p没有左子树的充要条件是(  B)。

    A. p->lc=NULL                 B. p->ltag=1  

    C. p->ltag=1 且p->lc=NULL    D. 以上都不对  

    9. 设n , m 为一棵二叉树上的两个结点,在中序遍历序列中n在m前的条件是(B)。

    A. n在m右方                 B. n在m 左方   

    C. n是m的祖先               D. n是m的子孙

    10. 如果F是由有序树T转换而来的二叉树,那么T中结点的前序就是F中结点的(B  )。

    A. 中序     B. 前序         C. 后序         D. 层次序//记住前序对应前序

    11. 欲实现任意二叉树的后序遍历的非递归算法而不必使用栈,最佳方案是二叉树采用(  A)存储结构。

    A. 三叉链表   B. 广义表       C. 二叉链表    D. 顺序

    12. 下面叙述正确的是(D)。

    A. 二叉树是特殊的树     //二叉树不是特殊的树

    B. 二叉树等价于度为2的树

    C. 完全二叉树必为满二叉树

    D. 二叉树的左右子树有次序之分

    13. 任何一棵二叉树的叶子结点在先序、中序和后序遍历序列中的相对次序(A )。

    A. 不发生改变                 B. 发生改变

    C. 不能确定                    D. 以上都不对

    14. 已知一棵完全二叉树的结点总数为9个,则最后一层的结点数为(B )。

    A. 1         B. 2           C. 3           D. 4

    15. 根据先序序列ABDC和中序序列DBAC确定对应的二叉树,该二叉树(  A  )。

    A. 是完全二叉树                             B. 不是完全二叉树

    C. 是满二叉树                                  D. 不是满二叉树

    二、判断题

    1. 二叉树中每个结点的度不能超过2,所以二叉树是一种特殊的树。                (× )

    2. 二叉树的前序遍历中,任意结点均处在其子女结点之前。                                 (√  )

    3. 线索二叉树是一种逻辑结构。                                          //存储结构              (×)

    4. 哈夫曼树的总结点个数(多于1时)不能为偶数。                                      (√)

    5. 由二叉树的先序序列和后序序列可以唯一确定一颗二叉树。                     (×)

    6. 树的后序遍历与其对应的二叉树的后序遍历序列相同。//正常情况是对应的(√)

    7. 根据任意一种遍历序列即可唯一确定对应的二叉树。                                          (√)

    8. 满二叉树也是完全二叉树。                                                                                (√)

    9. 哈夫曼树一定是完全二叉树。                                                                                    (×)

    10. 树的子树是无序的。//二叉树的子树有序                                                  (×)

    三、填空题

    1. 假定一棵树的广义表表示为A(B(E),C(F(H,I,J),G),D),则该树的度为_3____,树的深度为__4___,终端结点的个数为____6__,单分支结点的个数为__1____,双分支结点的个数为____1__,三分支结点的个数为____2___,C结点的双亲结点为___A____,其孩子结点为____F___和___G____结点。1. 3,4,6,1,1,2,A,F,G

    2. 设F是一个森林,B是由F转换得到的二叉树,F中有n个非终端结点,则B中右指针域为空的结点有____ n+1___个。//画一画

    3. 对于一个有n个结点的二叉树,当它为一棵____完全____二叉树时具有最小高度,即为_______,当它为一棵单支树具有____最大___高度,即为__ n _____。

    4. 由带权为3,9,6,2,5的5个叶子结点构成一棵哈夫曼树,则带权路径长度为__ 55_。

    5. 在一棵二叉排序树上按____中序___遍历得到的结点序列是一个有序序列。

    6. 对于一棵具有n个结点的二叉树,当进行链接存储时,其二叉链表中的指针域的总数为2n _______个,其中_____ n-1__个用于链接孩子结点,___ n+1____个空闲着。

    7. 在一棵二叉树中,度为0的结点个数为n0,度为2的结点个数为n2,则n0=__ n2+1____。

    8. 一棵深度为k的满二叉树的结点总数为____2k-1___,一棵深度为k的完全二叉树的结点总数的最小值为___2k-1__最大值为__2k-1____。

    9. 由三个结点构成的二叉树,共有__5__种不同的形态。

    10. 设高度为h的二叉树中只有度为0和度为2的结点,则此类二叉树中所包含的结点数至少为__ 2h-1__。//就是完全二叉树啊

    11. 一棵含有n个结点的k叉树,___单支树___形态达到最大深度,___完全二叉树_形态达到最小深度。

    12. 对于一棵具有n个结点的二叉树,若一个结点的编号为i(1≤i≤n),则它的左孩子结点的编号为____2i ____,右孩子结点的编号为____2i+1____,双亲结点的编号为__ëi/2û______。

    13. 对于一棵具有n个结点的二叉树,采用二叉链表存储时,链表中指针域的总数为_____ 2n ____个,其中________ n-1___个用于链接孩子结点,__________ n+1___个空闲着。

    14. 哈夫曼树是指____________带权路径长度最小_______________的二叉树。

    15. 空树是指______结点数为0__________________,最小的树是指_______________只有一个根结点的树________。

    16. 二叉树的链式存储结构有_______二叉链表_______和_________三叉链表______两种。

    17. 三叉链表比二叉链表多一个指向_________双亲结点_____的指针域。

    18. 线索是指_______________指向结点前驱和后继信息的指针__________。

    19. 线索链表中的rtag域值为__1___时,表示该结点无右孩子,此时__Rchild____域为指向该结点后继线索的指针。

    20. 本节中我们学习的树的存储结构有__孩子表示法_______、____双亲表示法_______和_______长子兄弟表示法____。

    四、应用题

    1. 已知一棵树边的集合为{<i,m>,<i,n>,<e,i>,<b,e>,<b,d>,<a,b>,<g,j>,<g,k>,<c,g>,<c,f>,<h,l>,<c,h>,<a,c>},请画出这棵树,并回答下列问题:

    (1)哪个是根结点?

    (2)哪些是叶子结点?

    (3)哪个是结点g的双亲?

    (4)哪些是结点g的祖先?

    (5)哪些是结点g的孩子?

    (6)哪些是结点e的孩子?

    (7)哪些是结点e的兄弟?哪些是结点f的兄弟?

    (8)结点b和n的层次号分别是什么?

    (9)树的深度是多少?

    (10)以结点c为根的子树深度是多少?

    1. 解答:

    根据给定的边确定的树如图5-15所示。

    其中根结点为a;

    叶子结点有:d、m、n、j、k、f、l;

    c是结点g的双亲;

    a、c是结点g的祖先;

    j、k是结点g的孩子;

    m、n是结点e的子孙;

    e是结点d的兄弟;

    g、h是结点f的兄弟;

    结点b和n的层次号分别是2和5;

    树的深度为5。

    2. 一棵度为2的树与一棵二叉树有何区别。

    2. 解答:

    度为2的树有两个分支,但分支没有左右之分;一棵二叉树也有两个分支,但有左右之分,左右子树不能交换。//二叉树有次序

    3. 试分别画出具有3个结点的树和二叉树的所有不同形态?

    3. 解答: 3个结点的树有两种形态,3个结点的二叉树有5种形态

     
       

    4. 已知用一维数组存放的一棵完全二叉树:ABCDEFGHIJKL,写出该二叉树的先序、中序和后序遍历序列。

    4. 解答: 

    先序序列:ABDHIEJKCFLG

    中序序列:HDIBJEKALFCG

    后序序列:HIDJKEBLFGCA

    5. 一棵深度为H的满k叉树有如下性质:第H层上的结点都是叶子结点,其余各层上每个结点都有k棵非空子树,如果按层次自上至下,从左到右顺序从1开始对全部结点编号,回答下列问题:

    (1)各层的结点数目是多少?

    (2)编号为n的结点的父结点如果存在,编号是多少?

    (3)编号为n的结点的第i个孩子结点如果存在,编号是多少?

    (4)编号为n的结点有右兄弟的条件是什么?其右兄弟的编号是多少?

    5. 解答:

    (1)第i层上的结点数目是mi-1

    (2)编号为n的结点的父结点如果存在,编号是((n-2)/m)+1。

    (3)编号为n的结点的第i个孩子结点如果存在,编号是(n-1)*m+i+1。

    (4)编号为n的结点有右兄弟的条件是(n-1)%m≠0。其右兄弟的编号是n+1。//心好累,这个题就先放过了

    6. 找出所有满足下列条件的二叉树:

    (1)它们在先序遍历和中序遍历时,得到的遍历序列相同;

    (2)它们在后序遍历和中序遍历时,得到的遍历序列相同;

    (3)它们在先序遍历和后序遍历时,得到的遍历序列相同;

    6. 解答:

    (1)先序序列和中序序列相同的二叉树为:空树或者任一结点均无左孩子的非空二叉树;

    (2)中序序列和后序序列相同的二叉树为:空树或者任一结点均无右孩子的非空二叉树;

    (3)先序序列和后序序列相同的二叉树为:空树或仅有一个结点的二叉树。

    7. 假设一棵二叉树的先序序列为EBADCFHGIKJ,中序序列为ABCDEFGHIJK,请写出该二叉树的后序遍历序列。

    7. 解答:后序序列:ACDBGJKIHFE

    8. 假设一棵二叉树的后序序列为DCEGBFHKJIA,中序序列为DCBGEAHFIJK,请写出该二叉树的后序遍历序列。

    8. 解答:先序序列:ABCDGEIHFJK

    9. 给出如图5-14所示的森林的先根、后根遍历结点序列,然后画出该森林对应的二叉树。

    9. 解答:

    先根遍历:ABCDEFGHIJKLMNO   //不用空格,一棵树遍历完了之后再遍历另一颗

    后根遍历:BDEFCAHJIGKNOML

    森林转换成二叉树如图5-16所示。

    10.给定一组权值(5,9,11,2,7,16),试设计相应的哈夫曼树。

    10. 解答:构造而成的哈夫曼树如图5-17所示。

     

     //终于写对哈夫曼树,心塞塞,小姑娘的少女心啊

    五、算法设计题

    1. 一棵具有n个结点的完全二叉树以一维数组作为存储结构,试设计一个对该完全二叉树进行先序遍历的算法。

    1. 解答:这个问题可以用递归算法,也可用非递归算法,下面给出的为非递归算法。假设该完全二叉树的结点以层次为序,按照从上到下,同层从左到右顺序编号,存放在一个一维数组R[1..n]中,且用一个有足够大容量为maxlen的顺序栈作辅助存储,算法描述如下:

    preorder (R)  //先序遍历二叉树R

    int R[n];

    {  int root;

    SqStack *s;  //s为一个指针栈,类型为seqstack,其中包含top域和数组data

    s->top= -1;  //s栈置空

    root=1;

    while ((root<=n) && (s->top>-1))

    {  while (root<=n)

    {  printf(R[root]);

                 s->top++;

                     s->data[s->top]=root;

                     root=2*root;

    }

             if (s->top>-1)  //栈非空访问,遍历右子树

             {  root=s->data[s->top]*2+1;

                s->top--;

    }

    }

    }

    2. 给定一棵用二叉链表表示的二叉树,其中的指针t指向根结点,试写出从根开始,按层次遍历二叉树的算法,同层的结点按从左至右的次序访问。

    2. 解答:考虑用一个顺序队que来保存遍历过程中的各个结点,由于二叉树以二叉链表存储,所以可设que为一个指向数据类型为bitree的指针数组,最大容量为maxnum,下标从1开始,同层结点从左到右存放。算法中的front为队头指针,rear为队尾指针。

    levelorder (BiTree *t)  //按层次遍历二叉树t

    {  BiTree *que[maxnum];  

    int rear,front;

    if (t!=NULL)

    {  front=0;  //置空队列

    rear=1;

    que[1]=t;

    do

             {  front=front%maxsize+1;  //出队

                t=que[front];

                printf(t->data);

                if (t->lchild!=NULL)  //左子树的根结点入队

                {  rear=rear%maxsize+1;

                       que[rear]=t->lchild;

    }

                if (t->rchild!=NULL)  //右子树的根结点入队

                {  rear=rear%maxsize+1;

                   que[rear]=t->rchild;

    }

    }while (rear= =front);  //队列为空时结束

    }

    }

    3. 写出在中序线索二叉树中结点P的右子树中插入一个结点s的算法。

    3. 解答:设该线索二叉树类型为bithptr,包含5个域:lchild,ltag,data,rchild,rtag。

    insert(p, s) //将s结点作为p的右子树插入

    BiThrNode *p,*s;

    {  BiThrNode *q;

    if (p->rtag= =1)  //无右子树,则有右线索

    {  s->rchild=p->rchild;

    s->rtag=1;

    p->rchild=s;

    p->rtag=0;

    }

    else

    {  q=p->rchild;

    while(q->ltag= =0)  //查找p所指结点中序后继,即为其右子树中最左下的结点

                 q=q->lchild;

    q->lchild=p->rchild;

    s->rtag=0;

    p->rchild=s;

    }

    s->lchild=p;  //将s结点的左线索指向p结点

    s->ltag=1;

    }

    4. 给定一棵二叉树,用二叉链表表示,其根指针为t,试写出求该二叉树中结点n的双亲结点的算法。若没有结点n或者该结点没有双亲结点,分别输出相应的信息;若结点n有双亲,输出其双亲的值。

    4. 解答:利用一个队列来完成,设该队列类型为指针类型,最大容量为maxnum。算法中的front为队头指针,rear为队尾指针,若当前队头结点的左、右子树的根结点不是所求结点,则将两子树的根结点入队,否则,队头指针所指结点即为结点的双亲。

    parentjudge(t,n)

    BiTree *t;

    int n;

    {  BiTree *que[maxnum];

    int front,rear;

    BiTree *parent;

    parent=NULL;

    if (t)

    if (t->data= =n)

    printf(“no parent!”);  //n是根结点,无双亲

    else

    {  front=0;  //初始化队列

                     rear=1;

                     que[1]=t;  //根结点进队

                     do

                     {  front=front%maxsize+1;

                        t=que[front];

                        if((t->lchild->data= =n)|| (t->rchild->data= =n))  //结点n有双亲

                        {  parent=t;

                           front=rear;

                           printf(“parent”,t->data);

    }

                        else

                        {  if (t->lchild!=NULL)  //左子树的根结点入队

                     {  rear=rear%maxsize+1;

                                que[rear]=t->lchild;

    }

                         if (t->rchild!=NULL)  //右子树的根结点入队

                         {  rear=rear%maxsize+1;

                            que[rear]=t->rchild;

    }

    }

    }while(rear= =front);  //队空时结束

               if (parent = =NULL)

                  printf(“结点不存在”);

    }

    }


  • 相关阅读:
    阶乘
    如何利用”七牛云”在UEditor实现图片的上传和浏览
    ueditor保存出现 从客户端(Note="<p>12345</p>")中检测到有潜在危险的 Request.Form 值
    配置进程外的Session
    数据库增加索引
    判断HTML中的checkbox是否被选中
    Oracle 配置文件目录
    Oracle 游标
    Oracle 分区表的索引、分区索引
    Oracle 索引
  • 原文地址:https://www.cnblogs.com/-XiangBei-/p/11143514.html
Copyright © 2011-2022 走看看