目录
1、什么是二叉搜索树?
2、手动模拟二叉搜索树
2.1、整体代码
2.2、查找数据
2.3、插入数据
2.4、删除数据
3、性能分析
1、什么是二叉搜索树?
二叉搜索树也叫作二叉排序树,可以使一颗空树,也可以是具有以下性质的二叉树:
- 若它的左子树不为空,则左子树的所节点的值都小于根节点的值
- 若它的右子树不为空,则右子树的所节点的值都大于根节点的值
- 它的左右子树也分别是二叉搜索树
举例:
2、手动模拟二叉搜索树
2.1、整体代码
public class BinarySearchTree {
static class TreeNode {
public int val;
public TreeNode left;
public TreeNode right;
public TreeNode(int val) {
this.val = val;
}
}
//搜索树的根
public TreeNode root;
/**
* 查找key是否存在于二叉搜索树当中
* @param key
* @return 当前节点找到返回地址,否则返回null
*/
public TreeNode search(int key) {
}
/**
* 插入节点
* @param key
*/
public boolean insert(int key) {
}
/**
* 删除节点
* @param key
*/
public void remove(int key) {
}
}
2.2、查找数据
代码:
/**
* 查找key是否存在于二叉搜索树当中
* @param key
* @return 当前节点找到返回地址,否则返回null
*/
public TreeNode search(int key) {
TreeNode cur = root;
while(cur != null) {
if(cur.val > key) {
cur = cur.left;
} else if(cur.val < key) {
cur = cur.right;
} else {
return cur;
}
}
return null;
}
2.3、插入数据
- 树为空,直接插入
- 树不为空,按照查找逻辑确定插入位置,如下:
代码:
/**
* 插入节点
* @param key
*/
public boolean insert(int key) {
TreeNode node = new TreeNode(key);
if(root == null) {
root = node;
return true;
}
TreeNode cur = root;
TreeNode parent = null;
while (cur != null) {
if(cur.val < key) {
parent = cur;
cur = cur.right;
} else if(cur.val > key) {
parent = cur;
cur = cur.left;
} else {
//相等,搜索树中,无法存放值相等的节点
return false;
}
}
//cur == null
if(key > parent.val) {
parent.right = node;
} else {
parent.left = node;
}
return true;
}
2.4、删除数据
设待删除结点为
cur,
待删除结点的双亲结点为
parent
1. cur.left == null
- cur 是 root,则 root = cur.right
- cur 不是 root,cur 是 parent.left,则 parent.left = cur.right
- cur 不是 root,cur 是 parent.right,则 parent.right = cur.right
2. cur.right == null
- cur 是 root,则 root = cur.left
- cur 不是 root,cur 是 parent.left,则 parent.left = cur.left
- cur 不是 root,cur 是 parent.right,则 parent.right = cur.left
3. cur.left != null && cur.right != null
- 需要使用替换法进行删除,即在它的右子树中寻找中序下的第一个结点(关键码最小),用它的值填补到被删除节点中,再来处理该结点的删除问题
代码:
/**
* 删除节点
* @param key
*/
public void remove(int key) {
TreeNode cur = root;
TreeNode parent = null;
while (cur != null) {
if(cur.val < key) {
parent = cur;
cur = cur.right;
} else if(cur.val > key) {
cur = cur.left;
} else {
//该节点需要删除
//1、cur.left == null
if(cur.left == null) {
if(cur == root) {
root = cur.right;
} else if(cur == parent.left) {
parent.left = cur.right;
} else {
parent.right = cur.right;
}
}//2、cur.right = null
else if(cur.right == null) {
if(cur == root) {
root = cur.left;
} else if(cur == parent.left) {
parent.left = cur.left;
} else {
parent.right = cur.left;
}
}
//3、cur.left != null && cur.right != null
else {
TreeNode targetParent = cur;
TreeNode target = cur.right;
//替换时,可以找左子树的最大值,也可以是右子树的最小值
//我这里是,左子树的最大值
while (target.left != null) {
targetParent = target;
target = target.left;
}
cur.val = target.val;
if(targetParent.left == target) {
targetParent.left = target.right;
} else {
targetParent.right = target.right;
}
}
}
}
}
3、性能分析
- 插入操作和删除操作都会必须先查找,查找效率代表了二叉搜索树中各个操作的性能
- 在二叉搜索树中,如果每个元素查找的概率相等,则二叉搜索树平均查找的长度是结点在二叉搜索树的深度的函数,即节点越多,比较的次数越多
- 对于同一个关键码集合,若插入次序不同,就可能会得到结构不同的二叉搜索树,如下:
这时,最优情况下,二叉搜索树为完全二叉树,平均比较次数为logN
最差情况下,二叉搜索树退化为单支树,平均比较次数为N/2
好啦,我们下期见咯~~~【AVL树】