数据结构与算法(九)
树结构基础部分
二叉树
为什么需要该数据结构
数组存储方式的分析
-
优点:
- 通过 下标 方式访问元素,速度快
- 对于 有序数组,还可以使用二分查找提高检索速度
-
缺点:如果要检索具体某个值或插入值(按一定顺序)会整体移动,效率较低,如下的示意图

链表存储方式的分析
优点:在一定程度上对数组存储方式有优化
例如:插入一个数值节点,只需要将插入节点,链接到链表中即可,同理,删除效率也很好

-
缺点:检索效率较低
需要从头结点开始遍历查找。
简单说:
- 数组访问快,增删慢
- 链表增删快,访问慢
那么就出现了 树 这种数据结构
树存储数据方式分析
提供数据 存储 、读取 效率。
例如:利用 二叉排序树(Binary Sort Tree),既可以保证数据的检索速度,同时也可以保证数据的插入、删除、修改 的速度

如图所示:
- 插入时,小的数在 左节点、大的数在 右节点
- 查找时:根据插入事的特性,基本上就类似折半查找了,每次都过滤一半的节点
- 删除时:只需要移动相邻的节点的引用
树的常用术语

-
节点:每一个圆圈表示一个节点,也称节点对象
-
根节点:最上面,最顶部的那个节点,也就是一棵树的入口
-
父节点:有子节点的节点
-
子节点
-
叶子节点:没有子节点的节点
-
节点的权:可以简单的理解为节点值
有时候也用 路径 来表示
-
路径:从 root 节点找到该节点的路线
-
层
-
子树:有子节点的父子两层就可以称为是一个子树
-
树的高度:最大层数
-
森林:多颗子树构成森林
二叉树的概念
-
树有很多种,每个节点 最多只能有两个子节点 的一种形式称为 二叉树
-
二叉树的子节点分为 左节点 和 右节点

-
如果该二叉树的所有 叶子节点 都在 最后一层,并且 节点总数 = 2^n-1 (n 为层数),则我们称为 满二叉树

-
如果该二叉树的所有叶子节点都在最 后一层或倒数第二层,而且 最后一层的叶子节点在左边连续,倒数第二层的叶子节点在右边连续,我们称为 完全二叉树

二叉树的遍历说明
有三种:
- 前序遍历:先输出父节点,再遍历左子树(递归)和右子树(递归)
- 中序遍历:先遍历左子树(递归),再输出父节点,再遍历右子树(递归)
- 后序遍历:先遍历左子树(递归),再遍历右子树(递归),最后输出父节点
前、中、后序主要区别是输出父节点的时间来区分
二叉树遍历思路分析

-
前序遍历:
- 先输出当前节点(初始节点是 root 节点)
- 如果左子节点不为空,则递归继续前序遍历
- 如果右子节点不为空,则递归继续前序遍历
上图的输出顺序为:1、2、3、4
-
中序遍历:
- 如果当前节点的左子节点不为空,则递归中序遍历
- 输出当前节点
- 如果当前节点的右子节点不为空,则递归中序
上图的输出顺序为:2、1、4、3
-
后序遍历:
- 如果左子节点不为空,则递归继续前序遍历
- 如果右子节点不为空,则递归继续前序遍历
- 输出当前节点
上图的输出顺序为:2、4、3、1
遍历代码实现
//定义节点
class HeroNode{
private int no;//当前节点编号
private String name;//当前节点名字
private HeroNode left;//当前节点左子树
private HeroNode right;//当前节点右子树
public HeroNode(int no, String name) {
this.no = no;
this.name = name;
}
public int getNo() {
return no;
}
public void setNo(int no) {
this.no = no;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public HeroNode getLeft() {
return left;
}
public void setLeft(HeroNode left) {
this.left = left;
}
public HeroNode getRight() {
return right;
}
public void setRight(HeroNode right) {
this.right = right;
}
@Override
public String toString() {
return "HeroNode{" +
"no=" + no +
", name='" + name + '\'' +
'}';
}
//前序遍历
public void preOrder(){
//先输出当前节点
System.out.println(this);
//递归遍历左子节点
if(this.left != null){
this.left.preOrder();
}
//递归遍历右子节点
if(this.right != null){
this.right.preOrder();
}
}
//中序遍历
public void infixOrder(){
//先输出左子树
if(this.left != null){
this.left.infixOrder();
}
//输出当前节点
System.out.println(this);
//输出右子树
if(this.right != null){
this.right.infixOrder();
}
}
//后序遍历
public void postOrder(){
//先输出左子树
if(this.left != null){
this.left.postOrder();
}
//输出右子树
if(this.right != null){
this.right.postOrder();
}
//输出当前节点
System.out.println(this);
}
}
//创建二叉树
class BinaryTree{
private HeroNode root;
public void setRoot(HeroNode root) {
this.root = root;
}
//前序遍历
public void preOrder(){
if(this.root != null){
this.root.preOrder();
}else{
System.out.println("当前二叉树为空");
}
}
//中序遍历
public void infixOrder(){
if(this.root != null){
this.root.infixOrder();
}else{
System.out.println("当前二叉树为空");
}
}
//后序遍历
public void postOrder(){
if(this.root != null){
this.root.postOrder();
}else{
System.out.println("当前二叉树为空");
}
}
}
//创建二叉树
BinaryTree binaryTree = new BinaryTree();
//手动创建二叉树
HeroNode root = new HeroNode(1,"宋江");
HeroNode hero1 = new HeroNode(2,"吴用");
HeroNode hero2 = new HeroNode(3,"卢俊义");
HeroNode hero3 = new HeroNode(4,"林冲");
root.setLeft(hero1);
root.setRight(hero2);
hero2.setRight(hero3);
binaryTree.setRoot(root);
//遍历二叉树
System.out.println("前序遍历");
binaryTree.preOrder();
//中序遍历
System.out.println("中序遍历");
binaryTree.infixOrder();
//后序遍历
System.out.println("后序遍历");
binaryTree.postOrder();

二叉树查找
要求:
- 编写前、中、后序查找方法
- 并分别使用三种查找方式,查找
id=5的节点 - 并分析各种查找方式,分别比较了多少次
由于二叉树的查找是遍历查找,所以就简单了,前面遍历规则已经写过了,改写成查找即可
//在节点中增加查找方法
//前序查找
public HeroNode preOrderSearch(int no){
if(this.no == no){
return this;
}
HeroNode resNode = null;
if(this.left != null){
resNode = this.left.preOrderSearch(no);
}
if(resNode != null){
return resNode;
}
if(this.right != null){
resNode = this.right.preOrderSearch(no);
}
return resNode;
}
//中序查找
public HeroNode infixOrderSearch(int no){
HeroNode resNode = null;
if(this.left != null){
resNode = this.left.infixOrderSearch(no);
}
if(resNode != null){
return resNode;
}
if(this.no == no){
return this;
}
if(this.right != null){
resNode = this.right.infixOrderSearch(no);
}
return resNode;
}
//后序查找
public HeroNode postOrderSearch(int no){
HeroNode resNode = null;
if(this.left != null){
resNode = this.left.postOrderSearch(no);
}
if(resNode != null){
return resNode;
}
if(this.right != null){
resNode = this.right.postOrderSearch(no);
}
if(resNode != null){
return resNode;
}
if(this.no == no){
return this;
}
return null;
}
//在二叉树中增加查找方法
//前序查找
public HeroNode preOrderSearch(int no){
HeroNode resNode = null;
if(this.root != null){
resNode = this.root.preOrderSearch(no);
}else{
System.out.println("当前二叉树为空");
}
return resNode;
}
//中序
public HeroNode infixOrderSearch(int no){
HeroNode resNode = null;
if(this.root != null){
resNode = this.root.infixOrderSearch(no);
}else{
System.out.println("当前二叉树为空");
}
return resNode;
}
//后序
public HeroNode postOrderSearch(int no){
HeroNode resNode = null;
if(this.root != null){
resNode = this.root.postOrderSearch(no);
}else{
System.out.println("当前二叉树为空");
}
return resNode;
}
可以看出:
- 找到的次数和 查找的顺序 有关,而查找顺序就是哪一序有关
- 找不到的次数则是相当于都遍历完成,所以是相等的次数
二叉树删除

要求:
- 如果删除的节点是 叶子节点,则删除该节点
- 如果删除的节点是非叶子节点,则删除该子树
测试:删除 5 号叶子节点和 3 号子树。
说明:目前的二叉树不是规则的,如果不删除子树,则需要考虑哪一个节点会被上提作为父节点。这个后续讲解排序二叉树时再来实现。先实现简单的
思路分析:
-
由于我们的二叉树是单向的
-
所以我们判定一个节点是否可以删除,是判断它的 子节点,是否可删除,否则则没法回到父节点删除了,因为要判断被删除的节点满足前面的两点要求
- 当前节点的 左子节点 不为空,并且左子节点就是要删除的节点,则 left = null,并且返回(结束递归删除)
- 当前节点的 右子节点 不为空,并且右子节点就是要删除的节点,则 right = null,并且返回(结束递归删除)
如果前面都没有删除,则继续递归删除。上面的要求是 2 点,实际上是,找到符合条件的节点则直接删除(因为不考虑是否有子树)
如果树只有一个 root 节点,则将 root 节点置空
//节点中增加删除方法
public void delete(int no){
if(this.left != null){
if(this.left.no == no){
this.left = null;
return;
}
}
if(this.right != null){
if(this.right.no == no){
this.right = null;
return;
}
}
//向左递归
if(this.left != null){
this.left.delete(no);
}
//向右递归
if(this.right != null){
this.right.delete(no);
}
}
//二叉树中增加删除方法
public void delete(int no){
if(this.root != null){
if(this.root.getNo() == no){
this.root = null;
return;
}else{
this.root.delete(no);
}
}else{
System.out.println("当前二叉树为空");
}
}
顺序存储二叉树
基本概念
从数据存储来看,数组存储 方式和 树 的存储方式可以 相互转换。即使数组可以转换成树,树也可以转换成数组。如下示意图

上图阅读说明:
- 圆圈顶部的数字对应了数组中的索引
- 圆圈内部的值对应的数数组元素的值
现在有两个要求:
- 上图的二叉树的节点,要求以数组的方式来存储
arr=[1,2,3,4,5,6,7] - 要求在遍历数组 arr 时,仍然可以以 前序、中序、后序的方式遍历
特点
想要 实现上面的两个要求,需要知道顺序存储二叉树的特点:
- 顺序二叉树 通常只考虑 完全二叉树
- 第 n 个元素的 左子节点 为
2*n+1 - 第 n 个元素的 右子节点 为
2*n+2 - 第 n 个元素的 父节点 为
(n-1)/2
注:n 表示二叉树中的第几个元素(按 0 开始编号)
比如:
- 元素 2 的左子节点为:
2 * 1 + 1 = 3,对比上图去查看,的确是 3 - 元素 2 的右子节点为:
2 * 1 + 2 = 4,也 就是元素 5 - 元素 3 的左子节点为:
2 * 2 + 1 = 5,也就是元素 6 - 元素 3 的父节点为:
(2-1)/2= 1/2 = 0,也就是根节点 1
三种遍历方式
//顺序存储二叉树
class ArrayBinaryTree{
private int[] array;
public ArrayBinaryTree(int[] array) {
this.array = array;
}
//前序遍历输出顺序存储二叉树
public void preOrder(int index){
if(array == null || array.length == 0){
System.out.println("数组为空,不能遍历完全二叉树");
}
//先输出当前元素
System.out.println(array[index]);
//再递归输出左子元素
if((index*2+1) < array.length){
preOrder((index*2 + 1));
}
//再递归输出右子元素
if((index*2+2) < array.length){
preOrder((index*2 +2));
}
}
public void preOrder(){
preOrder(0);
}
//中序遍历输出顺序存储二叉树
public void infixOrder(int index){
if(array == null || array.length == 0){
System.out.println("数组为空,不能遍历完全二叉树");
}
if((index*2 +1) < array.length){
infixOrder((index*2 + 1));
}
System.out.println(array[index]);
if((index*2 + 2) < array.length){
infixOrder((index*2+2));
}
}
public void infixOrder(){
infixOrder(0);
}
//后序遍历顺序存储二叉树
public void postOrder(int index){
if(array == null || array.length == 0){
System.out.println("数组为空,不能遍历完全二叉树");
}
if((index*2 + 1) < array.length){
postOrder((index*2+1));
}
if((index*2 + 2) < array.length){
postOrder((index*2+2));
}
System.out.println(array[index]);
}
public void postOrder(){
postOrder(0);
}
}



线索化二叉树
引出线索化二叉树
看如下问题:将数列 {1,3,6,8,10,14} 构成一颗二叉树

可以看到上图的二叉树为一颗 完全二叉树。对他进行分析,可以发现如下的一些问题:
- 当对上面的二叉树进行中序遍历时,数列为
8,3,10,1,14,6 - 但是
6,8,10,14这几个节点的左右指针,并没有完全用上
如果希望充分利用各个节点的左右指针,让各个节点可以 指向自己的前后节点,这个时候就可以使用 线索化二叉树 了
介绍
n 个节点的二叉树链表中含有 n + 1 个空指针域,他的推导公式为 2n-(n-1) = n + 1。
利用二叉链表中的空指针域,存放指向该节点在 某种遍历次序 *下的 **前驱** 和 **后继** 节点的指针,这种附加的指针称为*「线索」
- 前驱:一个节点的前一个节点
- 后继:一个节点的后一个节点
如下图,在中序遍历中,下图的中序遍历为 8,3,10,1,14,6,那么 8 的后继节点就为 3,3 的后继节点是 10

这种加上了线索的二叉树链表称为 线索链表(节点存储了下一个节点,组成了链表,并且一般的二叉树本来就是用链表实现的),相应的二叉树称为 线索二叉树(Threaded BinaryTree)。根据线索性质的不同,线索二叉树可分为:前、中、后序线索二叉树。
思路分析

将上图的二叉树,进行 中序线索二叉树,中序遍历的数列为 8,3,10,1,14,6。
那么以上图为例,线索化二叉树后的样子如下图

- 的后继节点为 3
- 3 由于 左右节点都有元素,不能线索化
- 10 的前驱节点为 3,后继节点为 1
- 1 不能线索化
- 14 的前驱节点为 1,后继节点为 6
- 6 有左节点,不能线索化
注意:当线索化二叉树后,那么一个 Node 节点的 left 和 right 属性,就有如下情况:
-
left 指向的是 左子树,也可能是指向 前驱节点
例如:节点 1 left 节点指向的是左子树,节点 10 的 left 指向的就是前驱节点
-
right 指向的是 右子树,也可能是指向 后继节点
例如:节点 3 的 right 指向的是右子树,节点 10 的 right 指向的是后继节点
代码实现
//创建节点
class HeroNode{
private int no;
private String name;
private HeroNode left;
private HeroNode right;
//左右子树的类型,如果是左右子树就是0,如果是前驱或者后继节点就是1
private int leftType;
private int rightType;
public HeroNode(int no, String name) {
this.no = no;
this.name = name;
}
public int getNo() {
return no;
}
public void setNo(int no) {
this.no = no;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public HeroNode getLeft() {
return left;
}
public void setLeft(HeroNode left) {
this.left = left;
}
public HeroNode getRight() {
return right;
}
public void setRight(HeroNode right) {
this.right = right;
}
public int getLeftType() {
return leftType;
}
public void setLeftType(int leftType) {
this.leftType = leftType;
}
public int getRightType() {
return rightType;
}
public void setRightType(int rightType) {
this.rightType = rightType;
}
@Override
public String toString() {
return "HeroNode{" +
"no=" + no +
", name='" + name + '\'' +
'}';
}
}
//创建线索化二叉树
class ThreadBinaryTree{
private HeroNode root;//表示跟节点
private HeroNode pre;//表示前驱节点
public void setRoot(HeroNode root) {
this.root = root;
}
public void threadedNodes(){
threadedNodes(root);
}
//中序线索化遍历
public void threadedList(){
if(root == null){
System.out.println("线索化二叉树为空");
return;
}
HeroNode node = root;
while(node != null){
//找到第一个有前驱节点的元素
while(node.getLeftType() != 1){
node = node.getLeft();
}
//输出第一个元素
System.out.println(node);
//当有后继节点的情况会进入
while(node.getRightType() == 1){
node = node.getRight();
System.out.println(node);
}
//当一个节点没有后继节点的时候直接获取右子树
node = node.getRight();
}
}
//中序线索化
public void threadedNodes(HeroNode node){
if(node == null){
return;
}
//先线索化左子树
threadedNodes(node.getLeft());
//再线索化当前节点
/*
* 1.设置前驱节点,
* 2.设置后继节点
* */
if(node.getLeft() == null){
node.setLeft(pre);
node.setLeftType(1);
}
//线索化后继节点,需要遍历到下一个节点的时候,通过下一个节点设置
if(pre != null && pre.getRight() == null){
pre.setRight(node);
pre.setRightType(1);
}
//将前驱节点后移
pre = node;
//线索化右子树
threadedNodes(node.getRight());
}
}
前序线索化和遍历
//前序线索化
public void preThreadedNodes(HeroNode node){
if(node == null){
return;
}
//先序列化自己
if(pre != null && node.getLeft() == null){
node.setLeft(pre);
node.setLeftType(1);
}
if(pre != null && pre.getRight() == null){
pre.setRight(node);
pre.setRightType(1);
}
pre = node;
//再序列化递归左子树
if(node.getLeftType() == 0){
preThreadedNodes(node.getLeft());
}
//再递归序列化右子树
if(node.getRightType() == 0){
preThreadedNodes(node.getRight());
}
}
//前序序列化遍历
public void preThreadedList(){
if(root == null){
System.out.println("线索化二叉树为空");
return;
}
HeroNode node = root;
while(node != null){
System.out.println(node);
while(node.getLeftType() == 0){
node = node.getLeft();
System.out.println(node);
}
while (node.getRightType() == 1) {
node = node.getRight();
System.out.println(node);
}
node = node.getRight();
}
}

浙公网安备 33010602011771号