• 首页 首页 icon
  • 工具库 工具库 icon
    • IP查询 IP查询 icon
  • 内容库 内容库 icon
    • 快讯库 快讯库 icon
    • 精品库 精品库 icon
    • 问答库 问答库 icon
  • 更多 更多 icon
    • 服务条款 服务条款 icon

数据结构入门指南二叉树

武飞扬头像
清水加冰
帮助1

前言

        在计算机科学中,数据结构是解决问题的关键。而二叉树作为最基本、最常用的数据结构之一,不仅在算法和数据处理中发挥着重要作用,也在日常生活中有着丰富的应用。无论是搜索引擎的索引算法、文件系统的组织方式,还是编译器的语法分析,二叉树都扮演着不可或缺的角色。为了便于大家更好的入门二叉树,本期先向大家简单介绍一下二叉树的基本性质,以及代码理解实现,给大家预预热。


 1. 树的概念及结构

   1.1 树的概念

         树是一种非线性的数据结构,它是由n(n>=0)个有限结点组成一个具有层次关系的集合。把它叫做树是因为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。

  • 有一个特殊的结点,称为根结点,根节点没有前驱结点
  • 除根节点外,其余结点被分成M(M>0)个互不相交的集合T1、T2、……、Tm,其中每一个集合Ti(1<= i<= m)又是一棵结构与树类似的子树。每棵子树的根结点有且只有一个前驱,可以有0个或多个后继
  • 因此,树是递归定义的。

学新通

 那这样是树还是非树?

 学新通

 答案是非树,树形结构中,子树之间不能有交集,否则就不是树形结构。

 1.2 树的基础概念

学新通

  • 节点的度:一个节点含有的子树的个数称为该节点的度; 如上图:A的为6
  • 叶节点或终端节点:度为0的节点称为叶节点; 如上图:B、C、H、I...等节点为叶节点
  • *非终端节点或分支节点:度不为0的节点; 如上图:D、E、F、G...等节点为分支节点
  • 双亲节点或父节点:若一个节点含有子节点,则这个节点称为其子节点的父节点; 如上图:A是B的父节点
  • 孩子节点或子节点:一个节点含有的子树的根节点称为该节点的子节点; 如上图:B是A的孩子节点
  • 兄弟节点:具有相同父节点的节点互称为兄弟节点; 如上图:B、C是兄弟节点
  • 树的度:一棵树中,最大的节点的度称为树的度; 如上图:树的度为6
  • *节点的层次:从根开始定义起,根为第1层,根的子节点为第2层,以此类推;
  • *树的高度或深度:树中节点的最大层次; 如上图:树的高度为4
  • *堂兄弟节点:双亲在同一层的节点互为堂兄弟;如上图:H、I互为兄弟节点
  • *节点的祖先:从根到该节点所经分支上的所有节点;如上图:A是所有节点的祖先
  • *子孙:以某节点为根的子树中任一节点都称为该节点的子孙。如上图:所有节点都是A的子孙
  • *森林:由m(m>0)棵互不相交的树的集合称为森林;

带星号的了解即可。

        这里我们重点说一下树的高度和节点层次,不同的数据结构书中一般有两种方式表示树的高度,一种是从0开始例如上述的树,根节点A高度就是0,到P、Q高度就是3。另外一种是从1开始,根节点1的高度为1,那P、Q的高度就是4。个人更推荐使用从1开始的,如果使用从0开始的,那如果是空树,在使用0表示就有点说不过去了,那空树的高度就只能是-1了。如果使用从1开始的,那空树就可以使用0来表示。

1.3 树的表示

         树结构相对线性表就比较复杂了,要存储表示起来就比较麻烦了,既然保存值域,也要保存结点和结点之间的关系,实际中树有很多种表示方式如:双亲表示法,孩子表示法、孩子双亲表示法以及孩子兄弟表示法等。

         首先在定义树的节点时就很为难,一个节点到底要定义多少个指向子节点的指针:

  1.  
    struct TreeNode
  2.  
    {
  3.  
    Datatype x;
  4.  
    struct TreeNode* child1;
  5.  
    struct TreeNode* child2;
  6.  
     
  7.  
    ……
  8.  
     
  9.  
    };

 要想定义一个节点就要先知道一个树的度,例如上述的树:

学新通

         这棵树的度为6,那我们定义时就要定义6个指针变量吗?那大部分的节点的子节点并没有达到6个,这样就会很浪费,定义也很费劲。

在C 中,有这样一种定义方法:

  1.  
    struct TreeNode
  2.  
    {
  3.  
    Datatype x;
  4.  
    vector<struct TreeNode*> childs;
  5.  
    };

         可以不规定个数,它使用数组的方式来存储,如果不够还可以进行增容,这种方式是使用顺序表来存储孩子节点的信息。

         还有一种非常巧的方式,叫孩子兄弟表示法,即左孩子右兄弟。

学新通

         拿这棵树为例,这种方法在定义节点时就只定义两个指针,一个指针叫左孩子指针,一个指针叫右兄弟指针。怎么指向呢?就是说无论一个节点有多少个孩子,它的孩子指针就只指向第一个孩子(最左边的孩子节点),剩下的孩子用第一个孩子的兄弟指针指向第二个,第二个孩子的兄弟指针指向第三个。

表示出来就是这样的结构:

学新通 除此之外还有其他的表示法,这里就不再一个一个地列举。

1.4 树的应用

 在现实生活中我们也经常使用到树状结构,例如:文件存储

学新通

 2. 二叉树

         了解完树的基本概念后,我们接下来进入二叉树的学习。

2.1 二叉树的概念

 一棵二叉树是结点的一个有限集合,该集合:

  1. 或者为空
  2. 由一个根节点加上两棵别称为左子树和右子树的二叉树组成

 二叉树的特点:

  1. 二叉树不存在度大于2的结点
  2. 二叉树的子树有左右之分,次序不能颠倒,因此二叉树是有序树

 对于任何一颗二叉树都是由一下几种情况复合而成。

学新通

         当然关于二叉树的基础概念还有很多,今天就先简单介绍,接下来给大家来点干货,先让大家切身体验一下二叉树。                                                

 2.2 二叉树的遍历

 当我们看到任何一颗二叉树都应该把它分为三个部分:

  • 根节点
  • 左子树
  • 右子树

学新通

 我们以这棵树为例进行分析:

学新通

         A为这棵树的根节点,B及其以下的节点(D、E)被称为左子树,C及其以下节点被称为右子树。然后B子树仍然可以分为D及其以下节点是左子树,E及其以下节点是右子树,然后再分,直到子节点为NULL,停止。

我们这里用的是分治算法。分治算法:

        把大问题分成类似的子问题,子问题再分成子问题,……,直到子问题无法再分割为止。

遍历可分为三种:

前序:也叫先根遍历,遍历顺序为:根、左子树、右子树

中序:也叫中根遍历,遍历顺序为:左子树、根、右子树

后序:也叫后根遍历,遍历顺序为:左子树、右子树、根

         我们先来尝试以下前序遍历,理解了前序遍历后两个就简单了。还是以这棵树为例:

学新通

         前序遍历,我们是先访问根,然后是左子树、右子树。那应该先遍历A,然后遍历A的左子树B(及其以下节点),然后就以B为根继续遍历它的左子树D(及其以下节点),然后再次以D为根开始,遍历左子树(NULL),然后开始返回到D,D再遍历右子树(NULL),然后D(这个子树)就遍历完毕,返回到B,B开始遍历右子树E(及其以下节点),E左子树为NULL返回到E,然后遍历右子树,右子树也为空(E子树遍历完毕),然后返回E(B的右子树遍历完毕),接着返回B(B的所有子树遍历完毕),接着返回到A,A开始遍历右子树……这个规律很符合递归。

        遍历完的顺序为:A、B、D、NULL、NULL、E、NULL、NULL、C、NULL、NULL。大部分学校讲的都是A、B、D、E、C。大部分同学应该都知道这个规律,但不知道为什么这样遍历。

        根据这个思路我们再来写一下中序遍历,中序遍历先访问左子树,然后是跟,最后是右子树。还是从A开始找,A的左子树B,然后以B为根,找B的左子树,接着以D为根,找D的左子树,D的左子树为NULL(D左子树遍历结束),返回到D(根),开始遍历D的右子树……

学新通

         最后遍历的顺序为:【NULL、D、NULL(B的左子树)、B(根)、NULL、E、NULL              (B的右子树)、】(A的左子树)、A(根)、【NULL、C、NULL】(A的右子树)。整理一下:

NULL、D、NULL、B、NULL、E、NULL、A、NULL、C、NULL(D、B、E、C、A)。

         我们接着写一下后序遍历:NULL、NULL、D、NULL、NULL、E、B、NULL、NULL、C、A(D、E、B、C、A)。

我们再来练一个:

学新通

         前序遍历:A(根)||这部分为整体二叉树的根、B(左子树)、NULL(B的左子树)、D(B的右子树)、F(D的左子树)、NULL(F的左子树)、NULL(F的右子树)、NULL(D的右子树)||这部分属于A的左子树、C、E、NULL(E的左子树)、NULL(E的右子树)、NULL(C的右子树)||这部分为A的右子树。

 后续的中序遍历和后续遍历大家可以自己私下练一下。

        好了我们已经了解了遍历,接下来我们来说实现以下前序遍历。给大家来点干货,便于大家更好理解。

 我们依然以这个简单的二叉树为例进行实现。

学新通

 我们先定义一个二叉树的节点:

  1.  
    typedef char Datatype;
  2.  
    typedef struct BinaryTreeNode
  3.  
    {
  4.  
    Datatype data;
  5.  
    struct BinaryTreeNode* left;
  6.  
    struct BinaryTreeNode* right;
  7.  
    }BTNode;

 然后就是它的前序遍历的实现:

  1.  
    void PrevOrder(BTNode* root)
  2.  
    {
  3.  
     
  4.  
    }

        我们知道前序遍历的顺序是根,然后是左子树,最后是右子树。因此在开始前我们先判断以下root是否为NULL,如果不为NULL我们就打印根节点的数据。 

  1.  
    void PrevOrder(BTNode* root)
  2.  
    {
  3.  
    if (root == NULL)
  4.  
    {
  5.  
    printf("NULL ");
  6.  
    return;
  7.  
    }
  8.  
    printf("%c ", root->data);
  9.  
     
  10.  
    }

         那如何遍历到左子树、右子树呢?其实很简单,我们之前介绍的时候说:二叉树的遍历复合递归结构,这里我们就可以使用递归来完成遍历,代码如下:

  1.  
    void PrevOrder(BTNode* root)
  2.  
    {
  3.  
    if (root == NULL)
  4.  
    {
  5.  
    printf("NULL ");
  6.  
    return;
  7.  
    }
  8.  
    printf("%c ", root->data);
  9.  
    PrevOrder(root->left);
  10.  
    PrevOrder(root->right);
  11.  
    }

         先遍历左子树,再遍历右子树,那调用的顺序就先调用自己传左孩子指针过去,以左孩子节点为根,再按照这个程序进行执行,再次传左孩子指针过去……,知道左孩子为NULL,返回上一层,开始遍历右子树。

 我们可以简单粗暴的测试以下,测试代码如下:

  1.  
    #include<stdio.h>
  2.  
    #include<stdlib.h>
  3.  
     
  4.  
    typedef char Datatype;
  5.  
    typedef struct BinaryTreeNode
  6.  
    {
  7.  
    Datatype data;
  8.  
    struct BinaryTreeNode* left;
  9.  
    struct BinaryTreeNode* right;
  10.  
    }BTNode;
  11.  
     
  12.  
    void PrevOrder(BTNode* root)
  13.  
    {
  14.  
    if (root == NULL)
  15.  
    {
  16.  
    printf("NULL ");
  17.  
    return;
  18.  
    }
  19.  
    printf("%c ", root->data);
  20.  
    PrevOrder(root->left);
  21.  
    PrevOrder(root->right);
  22.  
    }
  23.  
    int main()
  24.  
    {
  25.  
    BTNode* A = (BTNode*)malloc(sizeof(BTNode));
  26.  
    A->data = 'A';
  27.  
    A->left = NULL;
  28.  
    A->right = NULL;
  29.  
    BTNode* B = (BTNode*)malloc(sizeof(BTNode));
  30.  
    B->data = 'B';
  31.  
    B->left = NULL;
  32.  
    B->right = NULL;
  33.  
    BTNode* C = (BTNode*)malloc(sizeof(BTNode));
  34.  
    C->data = 'C';
  35.  
    C->left = NULL;
  36.  
    C->right = NULL;
  37.  
    BTNode* D = (BTNode*)malloc(sizeof(BTNode));
  38.  
    D->data = 'D';
  39.  
    D->left = NULL;
  40.  
    D->right = NULL;
  41.  
    BTNode* E = (BTNode*)malloc(sizeof(BTNode));
  42.  
    E->data = 'E';
  43.  
    E->left = NULL;
  44.  
    E->right = NULL;
  45.  
     
  46.  
    A->left = B;
  47.  
    A->right = C;
  48.  
    B->left = D;
  49.  
    B->right = E;
  50.  
     
  51.  
    PrevOrder(A);
  52.  
    printf("\n");
  53.  
    return 0;
  54.  
    }
学新通

执行结果:

学新通

 和上述分析的结果一致。

那剩下的中序遍历和后序遍历也很简单,只需要改变一下递归调用函数的次序即可:

  1.  
    //中序遍历
  2.  
    void InOrder(BTNode* root)
  3.  
    {
  4.  
    if (root == NULL)
  5.  
    {
  6.  
    printf("NULL ");
  7.  
    return;
  8.  
    }
  9.  
    InOrder(root->left);
  10.  
    printf("%c ", root->data);
  11.  
    InOrder(root->right);
  12.  
    }
  13.  
     
  14.  
    //后序遍历
  15.  
    void PostOrder(BTNode* root)
  16.  
    {
  17.  
    if (root == NULL)
  18.  
    {
  19.  
    printf("NULL ");
  20.  
    return;
  21.  
    }
  22.  
    PostOrder(root->left);
  23.  
    PostOrder(root->right);
  24.  
    printf("%c ", root->data);
  25.  
     
  26.  
    }
学新通

 我们也可以测试以下看看执行结果,测试代码如下:

  1.  
    typedef struct BinaryTreeNode
  2.  
    {
  3.  
    Datatype data;
  4.  
    struct BinaryTreeNode* left;
  5.  
    struct BinaryTreeNode* right;
  6.  
    }BTNode;
  7.  
     
  8.  
    void PrevOrder(BTNode* root)
  9.  
    {
  10.  
    if (root == NULL)
  11.  
    {
  12.  
    printf("NULL ");
  13.  
    return;
  14.  
    }
  15.  
    printf("%c ", root->data);
  16.  
    PrevOrder(root->left);
  17.  
    PrevOrder(root->right);
  18.  
    }
  19.  
    void InOrder(BTNode* root)
  20.  
    {
  21.  
    if (root == NULL)
  22.  
    {
  23.  
    printf("NULL ");
  24.  
    return;
  25.  
    }
  26.  
    InOrder(root->left);
  27.  
    printf("%c ", root->data);
  28.  
    InOrder(root->right);
  29.  
    }
  30.  
    void PostOrder(BTNode* root)
  31.  
    {
  32.  
    if (root == NULL)
  33.  
    {
  34.  
    printf("NULL ");
  35.  
    return;
  36.  
    }
  37.  
    PostOrder(root->left);
  38.  
    PostOrder(root->right);
  39.  
    printf("%c ", root->data);
  40.  
     
  41.  
    }
  42.  
    int main()
  43.  
    {
  44.  
    BTNode* A = (BTNode*)malloc(sizeof(BTNode));
  45.  
    A->data = 'A';
  46.  
    A->left = NULL;
  47.  
    A->right = NULL;
  48.  
    BTNode* B = (BTNode*)malloc(sizeof(BTNode));
  49.  
    B->data = 'B';
  50.  
    B->left = NULL;
  51.  
    B->right = NULL;
  52.  
    BTNode* C = (BTNode*)malloc(sizeof(BTNode));
  53.  
    C->data = 'C';
  54.  
    C->left = NULL;
  55.  
    C->right = NULL;
  56.  
    BTNode* D = (BTNode*)malloc(sizeof(BTNode));
  57.  
    D->data = 'D';
  58.  
    D->left = NULL;
  59.  
    D->right = NULL;
  60.  
    BTNode* E = (BTNode*)malloc(sizeof(BTNode));
  61.  
    E->data = 'E';
  62.  
    E->left = NULL;
  63.  
    E->right = NULL;
  64.  
     
  65.  
    A->left = B;
  66.  
    A->right = C;
  67.  
    B->left = D;
  68.  
    B->right = E;
  69.  
     
  70.  
    printf("前序遍历:");
  71.  
    PrevOrder(A);
  72.  
    printf("\n");
  73.  
     
  74.  
    printf("中序遍历:");
  75.  
    InOrder(A);
  76.  
    printf("\n");
  77.  
     
  78.  
    printf("后序遍历:");
  79.  
    PostOrder(A);
  80.  
    printf("\n");
  81.  
    return 0;
  82.  
    }
学新通

 执行结果如下:

学新通

 可以和上边的分析对比一下,没有问题,


总结

        二叉树遍历是学习和理解二叉树的重要部分。通过遍历,我们可以按照特定的顺序访问二叉树的节点,从而深入了解它们的结构和关系。在这篇博客中,我们介绍了三种常见的二叉树遍历方式:前序遍历、中序遍历和后序遍历,并对它们的原理、特点和应用进行了详细讨论。本期内容为预热阶段,先让大家熟悉一下二叉树,以便于后续二叉树的学习,好的本期内容到此结束,感谢阅读!

这篇好文章是转载于:学新通技术网

  • 版权申明: 本站部分内容来自互联网,仅供学习及演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,请提供相关证据及您的身份证明,我们将在收到邮件后48小时内删除。
  • 本站站名: 学新通技术网
  • 本文地址: /boutique/detail/tanhihabgh
系列文章
更多 icon
同类精品
更多 icon
继续加载