算法4-10:BST平衡二叉树的删除操作
偷懒方法
public void delete(Key key) {
insert(key, null);
}
这样的方法就是将key相应的值覆盖成null。当读取该键值的时候将会返回null。
这是一种偷懒的办法,可是在严肃的实际应用中肯定不能这样,一方面会造成内存浪费,还有一方面性能会越来越慢。
正规方法
先从简单的问题開始吧,怎样删掉BST中最小的键呢?
先找到最小的键,然后将右子节点挂在父节点上。
代码:
public void deleteMin() {
root = deleteMin(root);
}
private Node deleteMin(Node node) {
if(node.left != null) {
node.left = deleteMin(node.left);
node.count -= 1;
return node;
} else {
return node.right;
}
}
删除随意节点
删除随意节点的时候有三种情况能够考虑:
删除一个没有子节点的节点很easy了,仅仅要返回null就可以
删除一个仅仅有一个子节点的节点须要返回它唯一的一个子节点
最难的问题在于删除一个有两个子节点的节点。这时候就要将右子树中最小的节点分离出来。放在该节点原本的位置。这就是Hibbard删除法。
可是Hibbard删除法在使用一段时间后发现。整个树变得越来越不平衡。因此Hibbard删除法的平均复杂度是sqrtN。有人提出删除的时候随机取出左側或右側的继承节点。
代码:
public void delete(Key key) {
root = delete(root, key);
}
private Node delete(Node node, Key key) {
// 先定位到须要删除的点
if(node == null) return null;
int compare = key.compareTo(node.key);
if(compare < 0) {
node.left = delete(node.left, key);
updateCount(node);
return node;
} else if(compare > 0) {
node.right = delete(node.right, key);
updateCount(node);
return node;
}
// 已定位到相应的节点,開始删除。下面是没有子节点或仅仅有1个子节点的情况
if(node.left == null) return node.right;
if(node.right == null) return node.left;
// 有两个子节点时,採用Hibbard删除法,取出右側最小的节点代替被删除的节点
Node minNode = minNode(node.right);
node.right = deleteMin(node.right);
minNode.left = node.left;
minNode.right = node.right;
updateCount(minNode);
return minNode;
}
private void updateCount(Node node) {
node.count = 1+size(node.left)+size(node.right);
}
private Node minNode(Node root) {
if(root == null) return null;
Node node = root;
while(node.left != null) {
node = node.left;
}
return node;
}
兴许章节将会介绍红黑树,它能保证全部的操作都能保证logN复杂度。
算法4-10:BST平衡二叉树的删除操作的更多相关文章
- 平衡二叉树算法实现 c语言版 插入 删除
#include <stdio.h>#include <malloc.h>#include<stdlib.h> #define EQ(a,b) ((a)==(b)) ...
- 算法与数据结构(十一) 平衡二叉树(AVL树)
今天的博客是在上一篇博客的基础上进行的延伸.上一篇博客我们主要聊了二叉排序树,详情请戳<二叉排序树的查找.插入与删除>.本篇博客我们就在二叉排序树的基础上来聊聊平衡二叉树,也叫AVL树,A ...
- 算法与数据结构(十一) 平衡二叉树(AVL树)(Swift版)
今天的博客是在上一篇博客的基础上进行的延伸.上一篇博客我们主要聊了二叉排序树,详情请戳<二叉排序树的查找.插入与删除>.本篇博客我们就在二叉排序树的基础上来聊聊平衡二叉树,也叫AVL树,A ...
- MyBatis知多少(22)MyBatis删除操作
本节从表中使用MyBatis删除记录. 我们已经在MySQL下有EMPLOYEE表: CREATE TABLE EMPLOYEE ( id INT NOT NULL auto_increment, f ...
- ***Redis hash是一个string类型的field和value的映射表.它的添加、删除操作都是O(1)(平均)。hash特别适合用于存储对象
http://redis.readthedocs.org/en/latest/hash/hset.html HSET HSET key field value (存一个对象的时候key存) 将哈希 ...
- Java创建二叉搜索树,实现搜索,插入,删除操作
Java实现的二叉搜索树,并实现对该树的搜索,插入,删除操作(合并删除,复制删除) 首先我们要有一个编码的思路,大致如下: 1.查找:根据二叉搜索树的数据特点,我们可以根据节点的值得比较来实现查找,查 ...
- Java中list如何利用遍历进行删除操作
转: Java中list如何利用遍历进行删除操作 2018年03月31日 10:23:41 Little White_007 阅读数:3874 Java三种遍历如何进行list的便利删除: 1.f ...
- Golang map 如何进行删除操作?
Cyeam 关注 2017.11.02 10:02* 字数 372 阅读 2784评论 0喜欢 3 map 的删除操作 Golang 内置了哈希表,总体上是使用哈希链表实现的,如果出现哈希冲突,就把冲 ...
- ADO.NET 使用DELETE语句批量删除操作,提示超时,删除失败,几种优化解决思路
起因是如此简单的一句sql 提示:Timeout 时间已到.在操作完成之前超时时间已过或服务器未响应. 提供几种解决思路: 1.检查WHERE条件中字段是否已建索引 2.检查是否被其他表引用,引用表外 ...
随机推荐
- HDU 1087 Super Jumping! Jumping! Jumping! (LIS的最大和)
题意: 给定n个数的序列, 找出最长上升子序列和. 分析: #include<cstdio> #include<iostream> #include<queue> ...
- [第一波模拟\day3\T2]{独立集}(bubble.cpp)
[问题描述] 有一天,一个名叫顺旺基的程序员从石头里诞生了.又有一天,他学会了冒泡排序和独立集.在一个图里,独立集就是一个点集,满足任意两个点之间没有边.于是他就想把这两个东西结合在一起.众所周知,独 ...
- Mac OS X 上如何切换默认的 Python 版本?
- wps左侧显示目录
单击视图----文档结构图,在下拉选项中选择靠左即可,如图所示
- 71.mybatis 如何获取插入的id【从零开始学Spring Boot】
[从零开始学习Spirng Boot-常见异常汇总] 在之前的文章已经讲过spring boot集成mybatis了,但是忘记说一个很重要的知识点了,那就是获取获取主键id,这篇文章补充下,sprin ...
- The Grove(poj 3182)
题意:一个n*m(n,m<=50)的矩阵有一片连着的树林,Bessie要从起始位置出发绕林子一圈再回来,每次只能向横着.竖着或斜着走一步.问最少需多少步才能完成. /* 如果我们用搜索来写的话, ...
- Enchantress(hdu 3922)
首先考虑覆盖三个点的情况,有两种情况: ①:三个点都在圆上,则该圆是三角形的外接圆 ②:两个点在圆上,第三个点在圆内,且在圆上的两个点之间的线段一定是直径 如果是多个圆,就不停地迭代. 有一点重要的是 ...
- POJ3150:Cellular Automaton
题意看不懂加题目想不通,很菜. n<=500个数围城环,每次操作对每个数Ai把与i在环上相距不超过d<n/2(包括Ai)的数加起来取模m<=1e6,求K<=1e7次操作后的环. ...
- vim—基本命令1
---------------------------------------------------------------2015.07.27 :b 1 -> 切换到当前缓冲区 :2 4 ...
- hdu 5188 zhx and contest [ 排序 + 背包 ]
传送门 zhx and contest Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Othe ...