数组的优点与缺点
- 优点
通过下标方式访问元素,速度快。对于有序数组,还可使用二分查找提高检索速度。
2.缺点
如果要检索具体某个值,或者插入值(按一定顺序)会整体移动,效率较低
链式存储的优点和缺点
1.优点:
在一定程度上对数组存储方式有优化(比如:插入一个数值节点,只需要将插入节点,链接到链表中即可)
2.缺点:
在进行检索时,效率仍然较低,比如(检索某个值,需要从头节点开始遍历)
树存储方式分析
能提高数据存储,读取的效率, 比如利用 二叉排序树(Binary Sort Tree),既可以保证数据的检索速度,同时也可以保证数据的插入,删除,修改的速度。
人们更容易理解线性的数据结构(数组, 链表等)而不是树和图这种数据结构。树是非线性数据结构。它们不以线性方式存储数据。他们按层次组织数据.
树(英语:tree)是一种抽象数据类型(ADT: Abstract Data Type)或是实现这种抽象数据类型的数据结构,用来模拟具有树状结构性质的数据集合。它是由n(n>0)个有限节点组成一个具有层次关系的集合。把它叫做“树”是因为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。
1.每个节点都只有有限个子节点或无子节点;
2.没有父节点的节点称为根节点;
3.每一个非根节点有且只有一个父节点;
4.除了根节点外,每个子节点可以分为多个不相交的子树;
5.树里面没有环路(cycle)
树中任意节点的子节点之间没有顺序关系,这种树称为无序树,也称为自由树
树中任意节点的子节点之间有顺序关系,这种树称为有序树
其中二叉树又分为: 完全二叉树,满二叉树,排序二叉树(二叉查找树(英语:Binary Search Tree)),平衡二叉树(AVL树)
二叉树每个加点最多有两个子树(即:不存在度大于2的节点),二叉树的子树有左右之分,次序不能颠倒
二叉树有5种基本形态:
1.二叉树可以是空集
2.可以有空的左子树
3.可以有空右子树;
4.左右子树皆为空。
5.左右子树都有
1, 第i层最多有个2^i-1个节点
2,深度为的二叉树的至多有个节点 (根节点的深度是 0 ),深度为的二叉树的至多有个节点 (根节点的深度是 0 )
- 非空的二叉树,若树叶总数为n0,分支度为2的总数为 n2,则n0=n2+1
层数为k, 有(2^k)-1个节点的二叉树就是满二叉树
1.第层的节点数量是: 2^k-1
2.总节点数: (2^k) -1
如果二叉树所有叶子节点都在最后一层或者倒数第二层,而且最后一层的叶子节点在左边连续,倒数第二层的叶子节点在右边连续 为完全二叉树
除最后一层外,若其余层都是满的,并且最后一层或者是满的,或者是在右边缺少连续若干节点
完全二叉树是效率很高的数据结构,堆是一种完全二叉树,所以效率极高. 在做堆排序的时候, 就是把数组看成一个完全二叉树来对待
AVL树得名于它的发明者G. M. Adelson-Velsky和Evgenii Landis,他们在1962年的论文《An algorithm for the organization of information》中公开了这一数据结构。
定义: 这种左右子树的高度相差不超过 1 的树为平衡二叉树。
平衡之意,如天平,即两边的分量大约相同。
二叉查找树(英语:Binary Search Tree),也称为二叉搜索树、有序二叉树(ordered binary tree)或排序二叉树(sorted binary tree),是指一棵空树或者具有下列性质的二叉树
1.若任意节点的左子树不空,则左子树上所有节点的值均小于它的根节点的值;
2.若任意节点的右子树不空,则右子树上所有节点的值均大于它的根节点的值;
3.任意节点的左、右子树也分别为二叉查找树;
没有键值相等的节点
二叉树有3中遍历方式: 先序遍历, 中序遍历,后序遍历,根据父节点先中后顺序依次分为三种遍历顺序
先访问根节点,然后访问左节点,最后访问右节点(根->左->右)
package com.zyd
object BinaryTreeDemo {
def main(args: Array[String]): Unit = {
//先用比较简单的方法,直接关联的方法
val root1 = new HeroNode(1, "a1")
val root2 = new HeroNode(2, "b2")
val root3 = new HeroNode(3, "c3")
val root4 = new HeroNode(4, "d4")
val root5 = new HeroNode(5, "e5")
root1.left = root2
root1.right = root3
root3.left = root5
root3.right = root4
println("========前序遍历======")
root1.preOrder()
println("========中序遍历======")
root1.infixOrder()
println("========后序遍历======")
root1.postOrder()
}
}
// 定义节点
class HeroNode(hNo: Int, hName: String) {
val no = hNo
val name = hName
var left: HeroNode = null
var right: HeroNode = null
//前序遍历
def preOrder(): Unit = {
//先输出当前节点值
printf("节点信息: no=%d name=%s \n", no, name)
//向左遍历输出左子树
if (this.left != null) {
this.left.preOrder()
}
//向右边递归输出右子树
if (this.right != null) {
this.right.preOrder()
}
}
//中序遍历
def infixOrder(): Unit ={
//向左遍历输出左子树
if (this.left != null) {
this.left.infixOrder()
}
//输出当前节点值
printf("节点信息: no=%d name=%s \n", no, name)
//向右边递归输出右子树
if (this.right != null) {
this.right.infixOrder()
}
}
//后序遍历
def postOrder(): Unit ={
//向左遍历输出左子树
if (this.left != null) {
this.left.postOrder()
}
//向右边递归输出右子树
if (this.right != null) {
this.right.postOrder()
}
//输出当前节点值
printf("节点信息: no=%d name=%s \n", no, name)
}
}
class BinaryTree {
var root: HeroNode = null
def preOrder(): Unit = {
if (root != null) {
root.preOrder()
} else {
println("当前二叉树为空,不能遍历!")
}
}
}
package com.zyd
object BinaryTreeDemo {
def main(args: Array[String]): Unit = {
//先用比较简单的方法,直接关联的方法
val root1 = new HeroNode(1, "a1")
val root2 = new HeroNode(2, "b2")
val root3 = new HeroNode(3, "c3")
val root4 = new HeroNode(4, "d4")
val root5 = new HeroNode(5, "e5")
root1.left = root2
root1.right = root3
root3.left = root5
root3.right = root4
println("========前序查找======")
val node = root1.preOrder(5)
if (null != node) {
printf("find 编号=%d name=%s", node.no, node.name)
println
} else {
println("not find ...")
}
println("========中序查找======")
val node2 = root1.infixOrder(5)
if (null != node2) {
printf("find 编号=%d name=%s", node2.no, node2.name)
println
} else {
println("not find ...")
}
println("========后序查找======")
val node3 = root1.infixOrder(5)
if (null != node3) {
printf("find 编号=%d name=%s", node3.no, node3.name)
println
} else {
println("not find ...")
}
}
}
// 定义节点
class HeroNode(hNo: Int, hName: String) {
val no = hNo
val name = hName
var left: HeroNode = null
var right: HeroNode = null
//前序查找
def preOrder(no: Int): HeroNode = {
if (no == this.no) {
return this
}
//向左递归查找
var resNode: HeroNode = null
if (this.left != null) {
resNode = this.left.preOrder(no)
}
if (resNode != null) {
return resNode
}
//向右递归查找
if (this.right != null) {
resNode = this.right.preOrder(no)
}
return resNode
}
//中序查找
def infixOrder(no: Int): HeroNode = {
var resNode: HeroNode = null
//向左递归查找
if (this.left != null) {
resNode = this.left.infixOrder(no)
}
if (resNode != null) {
return resNode
}
if (no == this.no) {
return this
}
//向右递归查找
if (this.right != null) {
resNode = this.right.infixOrder(no)
}
return resNode
}
//后序遍历
def postOrder(no: Int): HeroNode = {
var resNode: HeroNode = null
if (this.left != null) {
resNode = this.left.postOrder(no)
}
if (resNode != null){
return resNode
}
if (this.right != null){
resNode = this.right.preOrder(no)
}
if (resNode != null){
return resNode
}
if (this.no == no){
return this
}
resNode
}
}
class BinaryTree {
var root: HeroNode = null
def preOrder(no: Int): HeroNode = {
if (root != null) {
root.preOrder(no)
} else {
println("当前二叉树为空,不能遍历!")
return null
}
}
def preOrderSearch(no: Int): HeroNode = {
if (root != null) {
return root.preOrder(no)
} else {
return null
}
}
def infixOrderSearch(no: Int): HeroNode = {
if (root != null) {
root.infixOrder(no)
} else {
println("当前二叉树为空,不能遍历!")
return null
}
}
def postOrderSearch(no: Int): HeroNode = {
if (root != null) {
root.postOrder(no)
} else {
println("当前二叉树为空,不能遍历!")
return null
}
}
}
(1) 如果删除的节点是叶子节点,删除该节点
(2)删除的节点是非叶子节点,则删除该子树
package com.zyd
object BinaryTreeDemo {
def main(args: Array[String]): Unit = {
//先用比较简单的方法,直接关联的方法
val root1 = new HeroNode(1, "a1")
val root2 = new HeroNode(2, "b2")
val root3 = new HeroNode(3, "c3")
val root4 = new HeroNode(4, "d4")
val root5 = new HeroNode(5, "e5")
root1.left = root2
root1.right = root3
root3.left = root5
root3.right = root4
}
}
// 定义节点
class HeroNode(hNo: Int, hName: String) {
val no = hNo
val name = hName
var left: HeroNode = null
var right: HeroNode = null
def delNode(no: Int): Unit = {
//首先比较当前节点的左子节点是否为要删除的节点
if (this.left != null && this.left.no == no) {
this.left = null
return
}
//比较当前节点的右子节点是否为要删除的节点
if (this.right != null && this.right.no == no) {
this.right = null
return
}
//向左递归删除
if (this.left != null) {
this.left.delNode(no)
}
//向右递归删除
if (this.right != null) {
this.right.delNode(no)
}
}
}
class BinaryTree {
var root: HeroNode = null
def delNode(no: Int): Unit = {
if (root != null) {
//先处理下root是不是删除的
if (root.no == no) {
root = null
} else {
root.delNode(no)
}
}
}
}
基本说明
对于数据存储,数组存储的方式可以看做树,树也可以看做数组存储
场景 给定 val arr = Array(1, 2, 3, 4, 5, 6) 实现前序遍历
package com.zyd
object ArrayTreeDemo {
def main(args: Array[String]): Unit = {
val arr = Array(1, 2, 3, 4, 5, 6)
val tree = new ArrayTree(arr)
tree.preOrder()
}
}
class ArrayTree(val arr: Array[Int]) {
def preOrder(): Unit = {
this.preOrder(0)
}
def preOrder(index: Int): Unit = {
if (arr == null || arr.length == 0) {
println("数组为空,不能按照二叉树前序遍历")
}
println(arr(index))
//向左递归遍历
if ((index * 2 + 1) < arr.length) {
preOrder(index * 2 + 1)
}
//向右递归遍历
if ((index * 2 + 2) < arr.length) {
preOrder(index * 2 + 2)
}
}
}
BST(Binary Sort Tree): 对于二叉排序树的任何一个非叶子节点,要求左子节点的值比当前节点的值小,右子节点的值比当前节点的值大
特别说明:相同的值可以放左边或右边
案例: 一个数组创建对应的二叉排序树,并使用中序遍历二叉排序树
package com.zyd
object Array2BinaryTree {
def main(args: Array[String]): Unit = {
val arr = Array(10, 5, 6, 1, 45, 3, 8)
val tee = new BinarySortTee
for (item <- arr){
tee.add(new Node(item))
}
//遍历二叉树
tee.infixOrder()
}
}
class Node(var value: Int) {
var left: Node = null
var right: Node = null
//添加方法
def add(node: Node): Unit = {
if (node == null) { //节点为空,返回
return
}
//如果插入的节点值小于当前节点的值
if (node.value < this.value) {
if (this.left == null) {
//说明该节点下没有左子节点
this.left = node
} else {
//递归进行插入
this.left.add(node)
}
} else {
//如果插入的节点值不小于当前节点的值
if (this.right == null) {
this.right = node
} else {
//递归进行插入
this.right.add(node)
}
}
}
//中序遍历
def infixOrder(): Unit = {
//向左递归遍历输出左子树
if (this.left != null) {
this.left.infixOrder()
}
//先输出当前节点值
printf("value=>"+ this.value+" ")
if (this.right != null) {
this.right.infixOrder()
}
}
}
//定义我们的二叉排序树
class BinarySortTee {
var root: Node = null
def add(node: Node): Unit = {
if (root == null) {
root = node
} else {
root.add(node)
}
}
//中序遍历
def infixOrder(): Unit = {
if (root != null) {
root.infixOrder()
} else {
println("当前二叉树为空,不能遍历")
}
}
}
结果
value=>1 value=>3 value=>5 value=>6 value=>8 value=>10 value=>45
1.检索要删除的节点,没找到就退出
2.检索到要删除的节点,需要找到该节点的父节点的叶子节点指向null
package com.zyd
object Array2BinaryTree {
def main(args: Array[String]): Unit = {
val arr = Array(7 ,3, 10, 12, 1, 9, 2)
val tee = new BinarySortTee
for (item <- arr) {
tee.add(new Node(item))
}
//遍历二叉树
tee.infixOrder()
tee.delNode(2)
println("delete....")
tee.infixOrder()
}
}
class Node(var value: Int) {
var left: Node = null
var right: Node = null
//添加方法
def add(node: Node): Unit = {
if (node == null) { //节点为空,返回
return
}
//如果插入的节点值小于当前节点的值
if (node.value < this.value) {
if (this.left == null) {
//说明该节点下没有左子节点
this.left = node
} else {
//递归进行插入
this.left.add(node)
}
} else {
//如果插入的节点值不小于当前节点的值
if (this.right == null) {
this.right = node
} else {
//递归进行插入
this.right.add(node)
}
}
}
//中序遍历
def infixOrder(): Unit = {
//向左递归遍历输出左子树
if (this.left != null) {
this.left.infixOrder()
}
//先输出当前节点值
printf("value=>" + this.value + " ")
if (this.right != null) {
this.right.infixOrder()
}
}
//根据值查找某个节点
def search(value: Int): Node = {
//先判断当前接待您是否为要删除的节点
if (value == this.value) {
return this
} else if (value < this.value) { //向左递归查找
if (this.left == null) {
return null
} else {
return this.left.search(value)
}
} else {
if (this.right == null) {
return null
} else {
return this.right.search(value)
}
}
}
//根据值找到某个节点的父节点
def searchParent(value: Int): Node = {
//1. 先判断当前节点的左子节点或右子节点是否是这个值
if ((this.left != null && this.left.value == value) ||
(this.right != null && this.right.value == value)
) {
return this
}
else {
//向左递归查找
if (this.left != null && value < this.value) {
return this.left.searchParent(value)
} else if (this.right != null && value > this.value) {
return this.right.searchParent(value)
} else {
return null
}
}
}
}
//定义我们的二叉排序树
class BinarySortTee {
var root: Node = null
def add(node: Node): Unit = {
if (root == null) {
root = node
} else {
root.add(node)
}
}
//中序遍历
def infixOrder(): Unit = {
if (root != null) {
root.infixOrder()
} else {
println("当前二叉树为空,不能遍历")
}
}
//查找节点
def search(value: Int): Node = {
if (root != null) {
return root.search(value)
} else {
return null
}
}
//查找父节点
def searchParent(value: Int): Node = {
if (root != null) {
return root.searchParent(value)
} else {
return null
}
}
//删除节点
def delNode(value: Int): Unit = {
if (root == null) {
//空树不删除
return null
}
//先看有没有要删除的节点
val targetNode = search(value)
if (targetNode == null) { //没有要删除的节点,直接返回
return
} else {
//查找 targetNode 的父节点
var parentNode = searchParent(value)
//叶子节点
if (targetNode.left == null && targetNode.right == null){
//判断targetNode 是 parentNode 的左子节点还是右子节点
if (parentNode.left != null && parentNode.left.value == value){
parentNode.left = null
}else{
parentNode.right = null
}
}
}
}
}
结果
value=>1 value=>2 value=>3 value=>7 value=>9 value=>10 value=>12
delete…
value=>1 value=>3 value=>7 value=>9 value=>10 value=>12
package com.zyd
object Array2BinaryTree {
def main(args: Array[String]): Unit = {
val arr = Array(7, 3, 10, 12, 1, 9, 2)
val tee = new BinarySortTee
for (item <- arr) {
tee.add(new Node(item))
}
//遍历二叉树
tee.infixOrder()
tee.delNode(1)
println("\ndelete....")
tee.infixOrder()
}
}
class Node(var value: Int) {
var left: Node = null
var right: Node = null
//添加方法
def add(node: Node): Unit = {
if (node == null) { //节点为空,返回
return
}
//如果插入的节点值小于当前节点的值
if (node.value < this.value) {
if (this.left == null) {
//说明该节点下没有左子节点
this.left = node
} else {
//递归进行插入
this.left.add(node)
}
} else {
//如果插入的节点值不小于当前节点的值
if (this.right == null) {
this.right = node
} else {
//递归进行插入
this.right.add(node)
}
}
}
//中序遍历
def infixOrder(): Unit = {
//向左递归遍历输出左子树
if (this.left != null) {
this.left.infixOrder()
}
//先输出当前节点值
printf("value=>" + this.value + " ")
if (this.right != null) {
this.right.infixOrder()
}
}
//根据值查找某个节点
def search(value: Int): Node = {
//先判断当前接待您是否为要删除的节点
if (value == this.value) {
return this
} else if (value < this.value) { //向左递归查找
if (this.left == null) {
return null
} else {
return this.left.search(value)
}
} else {
if (this.right == null) {
return null
} else {
return this.right.search(value)
}
}
}
//根据值找到某个节点的父节点
def searchParent(value: Int): Node = {
//1. 先判断当前节点的左子节点或右子节点是否是这个值
if ((this.left != null && this.left.value == value) ||
(this.right != null && this.right.value == value)
) {
return this
}
else {
//向左递归查找
if (this.left != null && value < this.value) {
return this.left.searchParent(value)
} else if (this.right != null && value > this.value) {
return this.right.searchParent(value)
} else {
return null
}
}
}
}
//定义我们的二叉排序树
class BinarySortTee {
var root: Node = null
def add(node: Node): Unit = {
if (root == null) {
root = node
} else {
root.add(node)
}
}
//中序遍历
def infixOrder(): Unit = {
if (root != null) {
root.infixOrder()
} else {
println("当前二叉树为空,不能遍历")
}
}
//查找节点
def search(value: Int): Node = {
if (root != null) {
return root.search(value)
} else {
return null
}
}
//查找父节点
def searchParent(value: Int): Node = {
if (root != null) {
return root.searchParent(value)
} else {
return null
}
}
//删除节点
def delNode(value: Int): Unit = {
if (root == null) {
//空树不删除
return null
}
//先看有没有要删除的节点
val targetNode = search(value)
if (targetNode == null) { //没有要删除的节点,直接返回
return
} else {
//查找 targetNode 的父节点
var parentNode = searchParent(value)
//叶子节点
if (targetNode.left == null && targetNode.right == null) {
//判断targetNode 是 parentNode 的左子节点还是右子节点
if (parentNode.left != null && parentNode.left.value == value) {
parentNode.left = null
} else {
parentNode.right = null
}
} else if (targetNode.left != null && targetNode.right != null){
//targetNode 只有两个节点
}else {
//targetNode 只有一个节点
//判断targetNode是parentNode的左子节点还是右子节点
if (targetNode.left != null){ //要删除的节点的左子节点不为空
//判断targetNode 是parentNode 的左还是右
if (parentNode.left.value == value){
parentNode.left = targetNode.left
}else{
parentNode.right = targetNode.left
}
} else { //目标节点右子节点不为空
if (parentNode.left.value == value){
parentNode.left = targetNode.right
}else{
parentNode.right = targetNode.right
}
}
}
}
}
}
思路:找到要删除节点的右子树里的最小值,删除其最小值并替换到当前要删除的值上
package com.zyd
object Array2BinaryTree {
def main(args: Array[String]): Unit = {
val arr = Array(7, 3, 10, 12, 1, 9, 2)
val tee = new BinarySortTee
for (item <- arr) {
tee.add(new Node(item))
}
//遍历二叉树
tee.infixOrder()
tee.delNode(7)
println("\ndelete....")
tee.infixOrder()
}
}
class Node(var value: Int) {
var left: Node = null
var right: Node = null
//添加方法
def add(node: Node): Unit = {
if (node == null) { //节点为空,返回
return
}
//如果插入的节点值小于当前节点的值
if (node.value < this.value) {
if (this.left == null) {
//说明该节点下没有左子节点
this.left = node
} else {
//递归进行插入
this.left.add(node)
}
} else {
//如果插入的节点值不小于当前节点的值
if (this.right == null) {
this.right = node
} else {
//递归进行插入
this.right.add(node)
}
}
}
//中序遍历
def infixOrder(): Unit = {
//向左递归遍历输出左子树
if (this.left != null) {
this.left.infixOrder()
}
//先输出当前节点值
printf("value=>" + this.value + " ")
if (this.right != null) {
this.right.infixOrder()
}
}
//根据值查找某个节点
def search(value: Int): Node = {
//先判断当前接待您是否为要删除的节点
if (value == this.value) {
return this
} else if (value < this.value) { //向左递归查找
if (this.left == null) {
return null
} else {
return this.left.search(value)
}
} else {
if (this.right == null) {
return null
} else {
return this.right.search(value)
}
}
}
//根据值找到某个节点的父节点
def searchParent(value: Int): Node = {
//1. 先判断当前节点的左子节点或右子节点是否是这个值
if ((this.left != null && this.left.value == value) ||
(this.right != null && this.right.value == value)
) {
return this
}
else {
//向左递归查找
if (this.left != null && value < this.value) {
return this.left.searchParent(value)
} else if (this.right != null && value > this.value) {
return this.right.searchParent(value)
} else {
return null
}
}
}
}
//定义我们的二叉排序树
class BinarySortTee {
var root: Node = null
def add(node: Node): Unit = {
if (root == null) {
root = node
} else {
root.add(node)
}
}
//中序遍历
def infixOrder(): Unit = {
if (root != null) {
root.infixOrder()
} else {
println("当前二叉树为空,不能遍历")
}
}
//查找节点
def search(value: Int): Node = {
if (root != null) {
return root.search(value)
} else {
return null
}
}
//查找父节点
def searchParent(value: Int): Node = {
if (root != null) {
return root.searchParent(value)
} else {
return null
}
}
//删除节点
def delNode(value: Int): Unit = {
if (root == null) {
//空树不删除
return null
}
//先看有没有要删除的节点
val targetNode = search(value)
if (targetNode == null) { //没有要删除的节点,直接返回
return
} else {
//查找 targetNode 的父节点
var parentNode = searchParent(value)
//叶子节点
if (targetNode.left == null && targetNode.right == null) {
//判断targetNode 是 parentNode 的左子节点还是右子节点
if (parentNode.left != null && parentNode.left.value == value) {
parentNode.left = null
} else {
parentNode.right = null
}
} else if (targetNode.left != null && targetNode.right != null) {
//targetNode 只有两个节点
val value = delRightTreeMin(targetNode.right)
targetNode.value = value
} else {
//targetNode 只有一个节点
//判断targetNode是parentNode的左子节点还是右子节点
if (targetNode.left != null) { //要删除的节点的左子节点不为空
//判断targetNode 是parentNode 的左还是右
if (parentNode.left.value == value) {
parentNode.left = targetNode.left
} else {
parentNode.right = targetNode.left
}
} else { //目标节点右子节点不为空
if (parentNode.left.value == value) {
parentNode.left = targetNode.right
} else {
parentNode.right = targetNode.right
}
}
}
}
}
def delRightTreeMin(node: Node): Int = {
var target = node
//使用while选混找到右子树的最小值
while (target.left != null) {
//为什么targetNode只遍历是left,如果left无值,right有值呢 答案:存储的二叉树是有序的,因此不存在right有,left无
target = target.left
}
val minValue = target.value
// 删除最小值对应的节点
delNode(minValue)
return minValue
}
}
package com.zyd
object Array2BinaryTree {
def main(args: Array[String]): Unit = {
val arr = Array(7, 3, 10, 12, 1, 9, 2)
val tee = new BinarySortTee
for (item <- arr) {
tee.add(new Node(item))
}
//遍历二叉树
tee.infixOrder()
tee.delNode(7)
println("\ndelete....")
tee.infixOrder()
}
}
class Node(var value: Int) {
var left: Node = null
var right: Node = null
//添加方法
def add(node: Node): Unit = {
if (node == null) { //节点为空,返回
return
}
//如果插入的节点值小于当前节点的值
if (node.value < this.value) {
if (this.left == null) {
//说明该节点下没有左子节点
this.left = node
} else {
//递归进行插入
this.left.add(node)
}
} else {
//如果插入的节点值不小于当前节点的值
if (this.right == null) {
this.right = node
} else {
//递归进行插入
this.right.add(node)
}
}
}
//中序遍历
def infixOrder(): Unit = {
//向左递归遍历输出左子树
if (this.left != null) {
this.left.infixOrder()
}
//先输出当前节点值
printf("value=>" + this.value + " ")
if (this.right != null) {
this.right.infixOrder()
}
}
//根据值查找某个节点
def search(value: Int): Node = {
//先判断当前接待您是否为要删除的节点
if (value == this.value) {
return this
} else if (value < this.value) { //向左递归查找
if (this.left == null) {
return null
} else {
return this.left.search(value)
}
} else {
if (this.right == null) {
return null
} else {
return this.right.search(value)
}
}
}
//根据值找到某个节点的父节点
def searchParent(value: Int): Node = {
//1. 先判断当前节点的左子节点或右子节点是否是这个值
if ((this.left != null && this.left.value == value) ||
(this.right != null && this.right.value == value)
) {
return this
}
else {
//向左递归查找
if (this.left != null && value < this.value) {
return this.left.searchParent(value)
} else if (this.right != null && value > this.value) {
return this.right.searchParent(value)
} else {
return null
}
}
}
}
//定义我们的二叉排序树
class BinarySortTee {
var root: Node = null
def add(node: Node): Unit = {
if (root == null) {
root = node
} else {
root.add(node)
}
}
//中序遍历
def infixOrder(): Unit = {
if (root != null) {
root.infixOrder()
} else {
println("当前二叉树为空,不能遍历")
}
}
//查找节点
def search(value: Int): Node = {
if (root != null) {
return root.search(value)
} else {
return null
}
}
//查找父节点
def searchParent(value: Int): Node = {
if (root != null) {
return root.searchParent(value)
} else {
return null
}
}
//删除节点
def delNode(value: Int): Unit = {
if (root == null) {
//空树不删除
return null
}
//先看有没有要删除的节点
val targetNode = search(value)
if (targetNode == null) { //没有要删除的节点,直接返回
return
} else {
//查找 targetNode 的父节点
var parentNode = searchParent(value)
//防止只有一个节点现象
if (parentNode == null){
root = null
return
}
//叶子节点
if (targetNode.left == null && targetNode.right == null) {
//判断targetNode 是 parentNode 的左子节点还是右子节点
if (parentNode.left != null && parentNode.left.value == value) {
parentNode.left = null
} else {
parentNode.right = null
}
} else if (targetNode.left != null && targetNode.right != null) {
//targetNode 只有两个节点
val value = delRightTreeMin(targetNode.right)
targetNode.value = value
} else {
//targetNode 只有一个节点
//判断targetNode是parentNode的左子节点还是右子节点
if (targetNode.left != null) { //要删除的节点的左子节点不为空
//判断targetNode 是parentNode 的左还是右
if (parentNode.left.value == value) {
parentNode.left = targetNode.left
} else {
parentNode.right = targetNode.left
}
} else { //目标节点右子节点不为空
if (parentNode.left.value == value) {
parentNode.left = targetNode.right
} else {
parentNode.right = targetNode.right
}
}
}
}
}
def delRightTreeMin(node: Node): Int = {
var target = node
//使用while选混找到右子树的最小值
while (target.left != null) {
//为什么targetNode只遍历是left,如果left无值,right有值呢 答案:存储的二叉树是有序的,因此不存在right有,left无
target = target.left
}
val minValue = target.value
// 删除最小值对应的节点
delNode(minValue)
return minValue
}
}