二叉树的相关定义及BST的实现
一、树的一些概念
树,子树,节点,叶子(终端节点),分支节点(分终端节点);
节点的度表示该节点拥有的子树个数,树的度是树内各节点度的最大值;
子节点(孩子),父节点(双亲),兄弟节点,祖先,子孙,堂兄弟,深度或高度;
森林是指若干棵或不相交的树,对于树中的每个节点,其子树的集合即为森林;
二叉树,满二叉树;
完全二叉树指一个深度为k的二叉树,它的每个节点的编号都与深度为k的满二叉树节点一一对应。例如:

而下图则不是完全二叉树

因为它的第六个节点对应满二叉树的第七个节点。
二、BST的实现及遍历
二叉树也有链式实现和顺序实现两种方式。顺序存储极大的浪费存储空间,除非是完全二叉树或者满二叉树。但是相比于链式结构同时也能够省下申请新节点的时间。而二叉搜索树Binary Search Tree(BST)几乎是最常见的应用,BST有可能是空树,否则它应该满足如下几点要求:
1. 若任意节点的左子树不空,则左子树上所有节点的值均小于它的根节点的值
2. 若任意节点的右子树不空,则右子树上所有节点的值均大于它的根节点的值
3. 任意节点的左、右子树也分别为二叉查找树
4. 没有键值相等的节点。
二叉搜索树使得查找搜索的复杂度降低为O(logn)【这里指的是期望,最差情况是O(n),退化成链表,即每个节点都比它的父亲节点要大或者小】
BST的节点定义:
public class Node{
private X data;
private Node lchild;
private Node rchild;
private Node parent;
public Node (X x)
{
data = x;
lchild = rchild = parent = null;
}
public X getData()
{
return data;
}
public Node getLchild()
{
return lchild;
}
//
public Node getRchild()
{
return rchild;
}
//
public Node getParent()
{
return parent;
}
}
这里有getter方法但是没有setter方法【data可以通过构造函数赋值(但无法改变)】,这里需要思考的一个事情是是否需要在类的外部对节点进行set操作。我觉得不合适,由于BST是有序的,data若要改直接破坏搜索树的顺序,显然是不可行的;设置子节点或者父节点更是直接破坏树的结构,因此不需要,反观get操作仅仅是获取该节点的信息,放开权限可能会使用户使用更加的方便,因为parent,data等成员域均为私有属性,在类外是无法访问的。
BST类声明及成员域
class BSTree <X extends Comparable<X>>{
private Node root;
/* Remaining code */
}
除了一个根节点外并没有其他的域,和链表一样的无情。这里值得注意的是泛型,这样的继承要求泛型类可比。而且只需要这一次对泛型的声明,其余类内代码拿来主义,可以直接使用泛型X,包括构造函数和成员函数,内部类等。
添加节点
public void add(X n)
{
Node m = new Node(n);
if (root == null)
{
root = m;
}
else
{
Node p = root;
while(p != null)
{
if ((p.data.compareTo(n)) > 0 && p.lchild != null)
p = p.lchild;
else if ((p.data.compareTo(n)) > 0 && p.lchild == null)
{
p.lchild = m;
m.parent = p;
break;
}
else if((p.data.compareTo(n)) <= 0 && p.rchild != null)
p = p.rchild;
else
{
p.rchild = m;
m.parent = p;
break;
}
}
}
}
基本思路是从根节点开始比较,如果该节点数据小于根节点那就查看其左节点,如果左节点为空直接插入,否则继续和左节点进行比较。循环如此,直到插入成功。
三序遍历
/*先序遍历*/
public void inOrder(Node t)
{
if (t != null)
{
System.out.print(t.getData());
inOrder(t.getLchild());
inOrder(t.getRchild());
}
}
/*中序遍历*/
public void preOrder(Node t)
{
if (t != null)
{
preOrder(t.getLchild());
System.out.print(t.getData());
preOrder(t.getRchild());
}
}
/*后序遍历*/
public void postOrder(Node t)
{
if (t != null)
{
postOrder(t.getLchild());
postOrder(t.getRchild());
System.out.print(t.getData());
}
}
先序遍历就是先访问根节点然后访问左子树,右子树(别搞错哦);中序遍历是先访问左子树,接着根和右子树;后序遍历是先访问右子树,接着根节点和左子树。显然递归能够很好的解决这个问题,比如中序遍历,建议直接对着代码思考,很清晰直观。
查找
public Node search (X key)
{
Node c = root;
while(c != null) {
if ((c.getData().compareTo(key)) == 0) {
return c;
} else if ((c.getData().compareTo(key)) > 0) {
c = c.lchild;
} else {
c = c.rchild;
}
}
System.out.println("search failed");
return null;
}
查找的思路和添加其实是一样的,不过更加简单,将判断子树是否为空改为是否和该节点数据相等即可。
遍历测试
BSTree <Integer> tree = new BSTree<>();
tree.add(3);
tree.add(1);
tree.add(5);
tree.add(2);
tree.add(4);
tree.add(6);
tree.inOrder(tree.getRoot());
System.out.println();
tree.preOrder(tree.getRoot());
System.out.println();
tree.postOrder(tree.getRoot());
System.out.println();
该树的结构如下所示:

先序遍历:312546
中序遍历:123456
后序遍历:214653
此外,一个比较重要且复杂的操作即删除节点,我们首先来看删除节点有几种情况。
第一种,该节点没有任何子节点,那么直接进行删除即可,不需影响任何其他节点,需要注意的是删除操作要判断这个节点是其父节点的左节点还是右节点。

比如删除节点66,直接从左图变为右图即可。无需改变结构。
第二种,该节点只有一个子节点,此时只需将该节点的父节点和子节点直接相连即可。

比如删除节点65,直接把70的左子节点连到66。
前两种比较简单,第三种情况是待删除节点拥有两个子树,这种情况稍微复杂,需要引入一个新的概念,即一个节点的后继节点。
一个节点后继节点的定义是在该节点的右子树上数值最小的节点,同样的,前驱节点的定义是其左子树的最大数值的节点。比如咱们例子中的节点60,它的后继节点就是65,而前驱节点是55。删除这种拥有两个子树的节点,需要首先交换该节点和其后继结点的位置(其实在代码中就是数据的交换),然后待删除的数据必将会转换为前两种情况之一,递归调用即可。

比如删除节点60,首先将该节点和其后继节点(65)交换数据【这时间是BST最危险的时刻,因为交换数据的原因已经不能保证顺序,所以务必要注意】,然后删除60所在的节点。
后继节点代码
public Node successor(Node s)
{
Node i = s.rchild;
while (i.lchild!=null)
{
i = i.lchild;
}
return i;
}
删除节点代码
public boolean remove(X r)
{
Node s = search(r);
if(s == null)
return false;
if (s == root)
{root = null;
return true;}
else if(s.lchild==null && s.rchild==null)
{
if (s.parent.lchild ==s) // s is leaf and left child of its parent s.parent.lchild = null; // delete s directly else s.parent.rchild = null; return true; } else if (s.rchild != null && s.lchild == null) //s只有左子节点
{
if (s.parent.lchild ==s) s.parent.lchild = s.rchild; else
s.parent.rchild = s.rchild; return true; } else if (s.rchild == null && s.lchild != null) //s只有右子节点
{
if (s.parent.lchild ==s) s.parent.lchild = s.lchild; else
s.parent.rchild = s.lchild; return true; } else //s有两个子节点
{
Node successor = successor(s);
X temp = successor.data;
remove(temp);
s.data = temp;
return true;
}
}
测试代码
BSTree <Double> tree = new BSTree<>();
tree.add(4.0);
tree.add(2.0);
tree.add(6.0);
tree.add(1.0);
tree.add(3.0);
tree.add(5.0);
tree.add(4.5);
tree.add(5.5);
tree.add(7.0);
tree.add(6.5);
tree.add(6.6);
tree.add(8.0);
System.out.print("删除前遍历结果为");
tree.preOrder(tree.getRoot());
System.out.println(); boolean b = tree.remove(6.0);
System.out.print("删除后遍历结果为");
tree.preOrder(tree.getRoot());
System.out.println();
测试结果
删除前遍历结果为1.0 2.0 3.0 4.0 4.5 5.0 5.5 6.0 6.5 6.6 7.0 8.0
删除后遍历结果为1.0 2.0 3.0 4.0 4.5 5.0 5.5 6.5 6.6 7.0 8.0
二叉树的相关定义及BST的实现的更多相关文章
- 二叉树各种相关操作(建立二叉树、前序、中序、后序、求二叉树的深度、查找二叉树节点,层次遍历二叉树等)(C语言版)
将二叉树相关的操作集中在一个实例里,有助于理解有关二叉树的相关操作: 1.定义树的结构体: typedef struct TreeNode{ int data; struct TreeNode *le ...
- Java 二叉树遍历相关操作
BST二叉搜索树节点定义: /** * BST树的节点类型 * @param <T> */ class BSTNode<T extends Comparable<T>&g ...
- Java实现二叉树及相关遍历方式
Java实现二叉树及相关遍历方式 在计算机科学中.二叉树是每一个节点最多有两个子树的树结构.通常子树被称作"左子树"(left subtree)和"右子树"(r ...
- 二叉树BinTree类定义
#include<iostream> using namespace std; template<class T> struct BinTreeNode{//二叉树结点类 T ...
- java实现二叉树的相关操作
import java.util.ArrayDeque; import java.util.Queue; public class CreateTree { /** * @param args */ ...
- JavaScript实现排序二叉树的相关算法
1.创建排序二叉树的构造函数 /** * 创建排序二叉树的构造函数 * @param valArr 排序二叉树中节点的值 * @constructor */ function BinaryTree(v ...
- 二叉树的相关在线编程(python)
问题一: 输入一个整数数组, 判断该数组是不是某二叉搜索树的后序遍历的结果.如果是则输出Yes,否则输出No. 假设输入的数组的任意两个数字都互不相同. 正确的后序遍历结果: sequence = [ ...
- OptimalSolution(2)--二叉树问题(2)BST、BBT、BSBT
一.判断二叉树是否为平衡二叉树(时间复杂度O(N)) 平衡二叉树就是:要么是一棵空树,要么任何一个节点的左右子树高度差的绝对值不超过1. 解法:整个过程为二叉树的后序遍历.对任何一个节点node来说, ...
- 模拟I2C协议学习点滴之程序相关定义
由于主机和从机都会给数据线SDA发信号,比如主机先给SDA发送数据后,从机收到数据后发送应答信号将SDA拉低,故SDA类型设定为inout.而DATA设定为inout类型,是起到校验通信的作用(后续的 ...
随机推荐
- Linux下部署FastDFS
FastDFS的安装 首先需要准备的资源如下: libfastcommon-master:https://github.com/happyfish100/libfastcommon FastD ...
- C# PC版微信消息监听自动回复
最近有个微商客户需要搞个 个人微信监听群消息关键字并实现自动回复功能, 因为他有很多群 很多买家咨询的话 一个个回复太麻烦, 客户要求 比如群里有人发 关键字 产品1 则自动回复产品1的相关描述 ...
- poj1905 Expanding Rods(二分)
题目链接:https://vjudge.net/problem/POJ-1905 题意:有一根长len的木棍,加热了n度,长度会膨胀为len*(1+n*c),c为膨胀系数.现在把这根木棍夹在两堵墙之间 ...
- java 模拟http请求,通过流(stream)的方式,发送json数据和文件
发送端: /** * 以流的方式 * 发送文件和json对象 * * @return */ public static String doPostFileStreamAndJsonObj(String ...
- L1-064 估值一亿的AI核心代码 (20 分)
L1-064 估值一亿的AI核心代码 (20 分) 以上图片来自新浪微博. 本题要求你实现一个稍微更值钱一点的 AI 英文问答程序,规则是: 无论用户说什么,首先把对方说的话在一行中原样打印出来: ...
- Git 入门:概念、原理、使用
出处: git入门:概念.原理.使用 git和Github 概念 Git --- 版本控制工具(命令). git是一个开源的分布式版本控制系统,用以有效.高速的处理从很小到非常大的项目版本管理.git ...
- react 深度 循环嵌套对象渲染问题 map
查了一些资料貌似react的循环渲染对象只有map,但map只支持数组对象. 接到后台数据如下 { "list": { "A": [{ "image& ...
- 美团2017年CodeM大赛-初赛B轮 黑白树 (树形dp)
大意: 给定树, 初始每个点全为白色, 点$i$有权值$k_i$, 表示选择$i$后, 所有距离$i$小于$k_i$的祖先(包括i)会变为黑色, 求最少选多少个点能使所有点变为黑色. 链上情况的话, ...
- css;js学习(一)
推荐基础前端学习地址https://ke.qq.com/course/315961蝉壳学院 清除浮动 .clearfix:before,.clearfix:after{ content: " ...
- 【原创】大数据基础之Chronos
官方:https://mesos.github.io/chronos/ mesos集群中替换crontab Chronos A fault tolerant job scheduler for Mes ...