算法与数据结构系列之[二叉树-上]

上篇从整体上介绍了树的一些基本概念,编程中我们用到的都是具体的树结构,比如二叉树。接下来我们用上、中、下三篇详细介绍二叉树,其中上篇为二叉树的理论部分,中篇为二叉树的C语言代码实现,下篇为二叉树的Java代码实现。

前言
现实中树的种类有很多种,数据结构中树的种类也不是单一的,有二叉树,B树,B+树,红黑树等,二叉树应该说是最常用的一种树形结构,这篇就重点介绍下二叉树。

一、二叉树的定义:
二叉树,顾名思义,就是每个节点最多有两个子树的树,分别是左子树和右子树,要注意这里用到“最多”这个词,就是说二叉树并不要求每个节点必须有两个子节点,有的节点只有左节点,有的节点只有右节点,同样叫做二叉树。
下图中列出几种二叉树:

图一

二、满二叉树和完全二叉树:
上图中有两个特殊的二叉树,分别是第二个和第三个。
其中第二个二叉树,叶子节点全部在最底层,除叶子节点外,每个节点都有左右两个子节点,这种二叉树叫做满二叉树。

上图第三个树,叶子节点在最下两层,最后一层的叶子节点向左连续排列,倒数第二层的叶子节点向右连续排列,这种二叉树叫做完全二叉树。

满二叉树比较好理解,完全二叉树有时容易混淆,下面这幅图列出几种非完全二叉树的情况:


图二

图二中的六棵二叉树都不是完全二叉树,上一排不满足条件一:最后一层叶子节点向左连续排列,下一排不满足条件二:倒数第二层的叶子节点向右连续排列,我们具体分析一下每棵树不满足条件的情况。

树1第三层5这个节点没有左子节点,从而导致最后一层的叶子节点从11到8向左不连续,加上5的左子节点10就连续了。树2第三层5这个节点没有子节点,导致从6的左子节点12到8不连续,缺少了5的左右子节点10和11,加上这两个节点就连续了。树3是第三层的4这个节点没有左子节点,导致从5的右子节点11向左不连续,也就是说最后一层最左边的叶子节点一定要有,否则就不是完全二叉树。

树4第二层的3这个节点没有子节点,导致倒数第二层没有叶子节点,从2的左子节点4向右不连续,最右边的叶子节点一定要有。树5第二层3这个节点只有左子节点6,没有右子节点,也不连续,和树4情况类似。树6第二层3这个节点只有右子节点7,没有左子节点,从而导致从4向右到7不连续。其实,要满足完全二叉树的条件,倒数第二层的节点个数要达到最大,也就是倒数第二层往上的每一层的每一个节点都要有两个子节点。所以完全二叉树的定义还可以是这样:叶子节点都在最底下两层,最后一层的叶子节点都靠左连续排列,并且除了最后一层,其他层的节点个数都要达到最大。

给图二中的每棵非完全二叉树适当加上一些节点,就可以转换为完全二叉树,如下图所示:

图三

三、二叉树的存储方式:

3.1、顺序存储结构
二叉树的顺序存储结构就是用一维数组存储二叉树中的节点,并且数组的下标要能够体现节点之间的逻辑关系,比如父节点与 子节点的关系,左右兄弟节点的关系。虽然二叉树的节点是以数组的方式存储,但要求能够以遍历树的方式遍历存储二叉树节点的数组,也就是可以用前序遍历,中序遍历,后序遍历的方式遍历数组。顺序存储结构只适合于完全二叉树,非完全二叉树并不适合用顺序存储结构,因为非完全二叉树会浪费比较多的存储空间,下面会用图说明这点的。
如下图,顺序结构图示:

图四

图四中前一棵二叉树的根节点存储在数组中下标为0的位置,此时数组中第n个元素的左子节点我2n+1,第n个元素的右节点为2n+2,第n个元素的父节点为(n-1)/2。

为了方便计算子节点,图四中后一棵二叉树的很节点存储在数组下标为1的位置,此时数组中第n个元素的左子节点为2n,第n个元素的右子节点为2n+1,第n个节点的父节点为n/2。

下图说明非完全二叉树顺序存储浪费存储空间的问题


图五

3.2、链式存储结构

顺序存储结构只适用于完全二叉树,大多数的情况还是要用到链式存储结构,可以用双向链表来存储二叉树的节点,这个双向量表包括一个数据域,两个指针域分别指向左子节点和右子节点,这样的链表称为二叉链表。

链式存储结构如下图所示:


图六

二叉链表的节点结构定义代码:

typedef struct BinaryTreeNode{
    TElemType data;
    struct BinaryTreeNode *left,*right;
}BinaryTreeNode,*BinaryTree;

四、二叉树的创建

要对二叉树进行操作,首先要创建一棵二叉树,接下来才可以对二叉树进行诸如遍历等操作,这里的创建二叉树代码只是创建二叉树的逻辑实现,并不是完善的代码,缺少相应的提示内容,并且不能单独执行,完善的代码将在后面具体的代码文章中列出。二叉树的创建和遍历都要用到递归的方法,对递归不熟练的小伙伴要自己补一下递归算法的相关内容。

void CreateBinaryTree(BinaryTree *T){
   TElemTpye val;
   scanf("%d",&val);
   if(val <= 0){    //val小于等于0都表示空树,这里是为了方便才这么写,其实0和负数是可以作为二叉树的节点的
       *T = NULL;
       return;
   }

   *T =(BinaryTree) malloc(sizeof(BinaryTreeNode));
   if(!*T)
       exit(-1);
   (*T)->data = val;
   CreateBinaryTree(&(*T)->left);
   CreateBinaryTree(&(*T)->right);
}

五、二叉树的遍历
二叉树的遍历是二叉树非常重要的操作,需要用到递归算法,经典的遍历方法有三种,分别是前序遍历、中序遍历、后序遍历。

  • 前序遍历:先输出父节点,再遍历左子树和右子树
  • 中序遍历:先遍历左子树,再输出父节点,再遍历右子树
  • 后序遍历:先遍历左子树,再遍历右子树,最后输出父节点

看输出父节点的顺序,确定是前序,中序还是后序遍历,输出父节点在先就是前序,输出父节点在中就是中序,输出父节点在后就是后序。

下面通过画图的方式详细描述这三种遍历方式:


图七

前序遍历代码:

void PreOrder(BinaryTree node){
    if(node == NULL)
        return;
    printf("%d",node->data);
    PreOrder(node->left);
    PreOrder(node->right);
}

中序遍历代码:

void InOrder(BinaryTree node){
    if(node == NULL)
        return;
    InOrder(node->left);
    printf("%d",node->data);
    InOrder(node->right);
}

后序遍历代码:

void PostOrder(BinaryTree node){
    if(node == NULL)
        return;
    PostOrder(node->left);
    PostOrder(node->right);
    printf("%d",node->data);
}

六、总结
这篇主要介绍了二叉树的定义,两种特殊的二叉树:满二叉树和完全二叉树,通过画图方式重点掌握完全二叉树的定义和辨别。接着介绍了二叉树的两种存储结构:顺序存储结构和链式存储结构,要注意顺序存储结构只适用于完全二叉树。最后介绍了二叉树的创建以及遍历方法,其中二叉树的三种遍历方法尤为重要,要掌握递归的内涵,能够利用递归方法遍历二叉树。其实二叉树还有一些不是特别重要的操作,这里不做赘述,将在后面完整的代码中体现。接下来的两篇将完整的C语言和Java代码贴出来,以供大家参考。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容