数据结构基础详解(C语言): 二叉树的遍历_线索二叉树_树的存储结构_树与森林详解

本文主要是介绍数据结构基础详解(C语言): 二叉树的遍历_线索二叉树_树的存储结构_树与森林详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文逻辑:
本文由二叉树的遍历起手,讲解了二叉树的三种遍历方式,以及如何构造一颗二叉树,并在此基础上,扩展了更好的二叉树-线索二叉树。树和森林的存储结构讲解中,重点就是将树与森林转换为二叉树,这样二叉树的手段就能使用到树与森林当中。最后,讲解了二叉树与森林的遍历。

文章目录

  • 1.二叉树的遍历
    • 1.1 二叉树的前,中,后序遍历
    • 1.2 二叉树的层次遍历
    • 1.3构造二叉树
  • 2.线索二叉树
    • 2.1 以中序线索二叉树为例,讲解线索二叉树
    • 2.2 二叉树的线索化
    • 2.3 线索二叉树中找前驱后继
      • 2.3.1 中序线索二叉树寻找前驱后继(重点)
      • 2.3.2 先序线索二叉树寻找前驱后继(了解)
      • 2.3.3 后序线索二叉树寻找前驱后继(了解)
  • 3.树的存储结构
    • 3.1 双亲表示法(顺序存储)
    • 3.2 孩子表示法(顺序+链式存储)
    • 3.3 孩子兄弟表示法(链式存储)->重点
    • 3.4 森林和二叉树的转换
  • 4.树的遍历
    • 4.1 树的先根遍历(先根,先根,就是先遍历根,再遍历子树)
    • 4.2 树的后根遍历
    • 4.3 树的层次遍历
  • 5.森林的遍历
    • 5.1 先序遍历森林
    • 5.2 中序遍历森林

1.二叉树的遍历

什么是遍历
遍历:按照某种次序把所有的结点都访问一遍
什么是层次遍历:基于树的层次特性确定的次序规则(从上到下,从左到右的遍历)

二叉树的递归特性:
1️⃣ 要么是个空二叉树
2️⃣ 要么就是由根节点+左子树+右子树 组成的二叉树

1.1 二叉树的前,中,后序遍历

二叉树有三种遍历的情形
1️⃣先序遍历:根左右
2️⃣中序遍历:左根右
3️⃣ 后序遍历:左右根
在这里插入图片描述

二叉树的遍历(手算)

给你一个二叉树,写出他的先序遍历,中序遍历,后序遍历,属于简单题
给出一个手算技巧,留出空,根据遍历规则,补全

1.2 二叉树的层次遍历

  • 算法思想
    1️⃣ 初始化一个辅助队列
    2️⃣ 根结点入队
    3️⃣ 若队列为空,则头结点出队。将其左孩子,右孩子插入队尾(如果有的话
    4️⃣ 重复3️⃣直至队列为空

1.3构造二叉树

直接说结论:
想要构造一个二叉树,必须知道的两种遍历序列,其中还必须有中序遍历。
也就是中序遍历+(前序遍历or后序遍历or层次遍历)

线索二叉树引入:

在二叉树的基础上,我们能否从一个指定结点开始中序遍历?
如何找到指定结点p在中序遍历序列中的前驱
如何找到p的中序后继
解决思路:
两个指针+目标结点p
从根节点出发,重新进行一次中序遍历,指针q记录当前访问的结点,指针pre记录上一个被访问的结点
当qp时,pre为前驱
当pre
p时,q为后继。
通过这种方法解决还是太麻烦了,所以引入线索二叉树

2.线索二叉树

2.1 以中序线索二叉树为例,讲解线索二叉树

n个结点的二叉树,有n+1个空链域,用来构成线索,记录前驱和后继

  • 前驱线索(由左孩子指针充当)
  • 后继线索(由右孩子指针充当)
    在这里插入图片描述

三种线索二叉树的对比
在这里插入图片描述

2.2 二叉树的线索化

为了区分指针是指向了孩子还是指向了前驱/后继
我们还需要两个标志 一个是ltag,一个是rtag,记录当前结点是否记录了线索

  • ltag=0,lchild指向左孩子
  • ltag=1,lchild指向前驱
  • rtag=0,lchild指向右孩子
  • rtag=1,lchild指向后继

一棵二叉树的中序线索化:
1.首先取得二叉树的中序序列,
假设是A4,A2,A5,A1,A6,A3
2.A4的前驱指针指向NULL,A3的后继指针指向NULL
3.遍历一遍树,如果左指针为空就指向前驱,如果右指针为空就指向后继。

2.3 线索二叉树中找前驱后继

2.3.1 中序线索二叉树寻找前驱后继(重点)

中序线索二叉树找中序后继

明确p点肯定有右孩子,没有右孩子的话,中序后继直接为NULL了
根据左根右,他的后继是右,也就是右子树,右孩子如果还有孩子,那继续左根右,即首先访问最左的点
综上所述,中序后继=p的右子树中最左下结点。

中序线索二叉树找中序前驱
有了之前的分析过程,很好推出,中序前驱=p的左子树中最右下结点

2.3.2 先序线索二叉树寻找前驱后继(了解)

先序线索二叉树找先序前驱
无法实现,除非土办法遍历或者,使用三叉链表(能找到p结点的父结点)
下面假设可以找到p的父节点.
在这里插入图片描述

先序线索二叉树找先序后继
先序后继就是最左的节点

2.3.3 后序线索二叉树寻找前驱后继(了解)

后序线索二叉树寻找后序前驱
p肯定有左孩子,若p有右孩子,则后序前驱是他的右孩子,没有就是左孩子

后序线索二叉树寻找后序后继
无法实现,除非土办法遍历或者,使用三叉链表(能找到p结点的父结点)
下面假设可以找到p的父节点.

在这里插入图片描述

3.树的存储结构

  • 双亲表示法(顺序存储)
  • 孩子表示法(顺序+链式存储)
  • ⭐️孩子兄弟表示法(链式存储)

3.1 双亲表示法(顺序存储)

核心:每个结点中保存指向双亲的指针
在这里插入图片描述

增删改查思路:
增加一个孩子,直接在顺序的数组中增加即可
删除一个孩子,会让某一块为空,可以让最后一个存储的孩子,覆盖上去,提高效率
查找的话:想要找到某个结点的孩子,比较麻烦,要全部遍历一遍树

优缺点:
优点:查指定结点的双亲很方便。
缺点:查指定结点的孩子只能从头遍历。

3.2 孩子表示法(顺序+链式存储)

核心:结点顺序存储,每一个结点的孩子,用链式存储
在这里插入图片描述

3.3 孩子兄弟表示法(链式存储)->重点

核心目的:将逻辑结构树能够转换为存储结构是二叉树的形式储存,这样我们就可以用处理二叉树的手段,处理它。

核心思路:左指针指向第一个孩子(最左边的孩子),右指针指向他的右兄弟

我们可以 轻松的将一个树改写成这种孩子兄弟表示法的形式
在这里插入图片描述

3.4 森林和二叉树的转换

在3.3的基础上,我们不难理解森林和二叉树的转换,就是按照孩子兄弟表示法
在这里插入图片描述

在这里插入图片描述

4.树的遍历

  • 树的先根遍历

4.1 树的先根遍历(先根,先根,就是先遍历根,再遍历子树)

核心:先根遍历。若树非空,先访问根结点,再依次对每棵子树进行先根遍历。

与二叉树遍历对比:树的先根遍历序列与二叉树的先序遍历相同

4.2 树的后根遍历

核心:后根遍历。若树非空,先依次对每棵子树进行后根遍历,最后访问根结点
与二叉树遍历对比:树的后根遍历序列与二叉树的中序遍历相同

4.3 树的层次遍历

跟二叉树的层次遍历一样,用队列实现

5.森林的遍历

森林是m棵互不相交的树的集合。每棵树去掉根节点,其各个子树又组成森林。

5.1 先序遍历森林

若森林为非空,则按如下规则进行遍历;
访问森林中第一棵树的根结点。
先序遍历第一棵树中根结点的子树森林
先序遍历除去第一棵之后剩余的树构成的森林。

树的遍历对比:效果等同于依次对各个树进行先根遍历

5.2 中序遍历森林

若森林为非空,则按如下规则进行遍历
中序遍历森林中第一棵树的根结点的子树森林
访问第一棵树的根结点。
中序遍历除去第一棵树之后剩余的树构成的森林。

树的遍历对比:效果等同于依次对各个树进行后根遍历

这篇关于数据结构基础详解(C语言): 二叉树的遍历_线索二叉树_树的存储结构_树与森林详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1143453

相关文章

Java使用SLF4J记录不同级别日志的示例详解

《Java使用SLF4J记录不同级别日志的示例详解》SLF4J是一个简单的日志门面,它允许在运行时选择不同的日志实现,这篇文章主要为大家详细介绍了如何使用SLF4J记录不同级别日志,感兴趣的可以了解下... 目录一、SLF4J简介二、添加依赖三、配置Logback四、记录不同级别的日志五、总结一、SLF4J

Java使用ANTLR4对Lua脚本语法校验详解

《Java使用ANTLR4对Lua脚本语法校验详解》ANTLR是一个强大的解析器生成器,用于读取、处理、执行或翻译结构化文本或二进制文件,下面就跟随小编一起看看Java如何使用ANTLR4对Lua脚本... 目录什么是ANTLR?第一个例子ANTLR4 的工作流程Lua脚本语法校验准备一个Lua Gramm

Android Mainline基础简介

《AndroidMainline基础简介》AndroidMainline是通过模块化更新Android核心组件的框架,可能提高安全性,本文给大家介绍AndroidMainline基础简介,感兴趣的朋... 目录关键要点什么是 android Mainline?Android Mainline 的工作原理关键

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字

Python列表去重的4种核心方法与实战指南详解

《Python列表去重的4种核心方法与实战指南详解》在Python开发中,处理列表数据时经常需要去除重复元素,本文将详细介绍4种最实用的列表去重方法,有需要的小伙伴可以根据自己的需要进行选择... 目录方法1:集合(set)去重法(最快速)方法2:顺序遍历法(保持顺序)方法3:副本删除法(原地修改)方法4:

python logging模块详解及其日志定时清理方式

《pythonlogging模块详解及其日志定时清理方式》:本文主要介绍pythonlogging模块详解及其日志定时清理方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录python logging模块及日志定时清理1.创建logger对象2.logging.basicCo

前端CSS Grid 布局示例详解

《前端CSSGrid布局示例详解》CSSGrid是一种二维布局系统,可以同时控制行和列,相比Flex(一维布局),更适合用在整体页面布局或复杂模块结构中,:本文主要介绍前端CSSGri... 目录css Grid 布局详解(通俗易懂版)一、概述二、基础概念三、创建 Grid 容器四、定义网格行和列五、设置行

Node.js 数据库 CRUD 项目示例详解(完美解决方案)

《Node.js数据库CRUD项目示例详解(完美解决方案)》:本文主要介绍Node.js数据库CRUD项目示例详解(完美解决方案),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考... 目录项目结构1. 初始化项目2. 配置数据库连接 (config/db.js)3. 创建模型 (models/

SQL表间关联查询实例详解

《SQL表间关联查询实例详解》本文主要讲解SQL语句中常用的表间关联查询方式,包括:左连接(leftjoin)、右连接(rightjoin)、全连接(fulljoin)、内连接(innerjoin)、... 目录简介样例准备左外连接右外连接全外连接内连接交叉连接自然连接简介本文主要讲解SQL语句中常用的表

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的