二叉查找树的懒惰删除(lazy deletion)
第四章习题:二叉查找树类实现懒惰删除,注意findMin()和findMax()(递归)
算是发布的第一篇学习笔记。也不敢保证写的代码一定正确,错了的地方请大家指正,谢谢。
直接开始吧。先谈谈数据结构,二叉查找树懒惰删除较于一般的二叉查找树,多了一些域:theSize(剩下的节点数)、deletedSize(懒惰删除的节点数)、BinaryNode<AnyType> min,max(用于保留在findMin和findMax方法中递归查询到的flag!=1的最值点);在内部节点类中,多了一个byte型的flag变量(=1则表示被删除)。在这里,也可以使用一个count域,这在有重复项时很常用,初始的count都是1,以后有插入就+1,有删除且count>0就-1.我使用的直接是flag标志变量。
class myLazyBinarySearchTree<AnyType extends Comparable<? super AnyType>> {
private BinaryNode<AnyType> root;
private int theSize;// 总节点数
private int deletedSize;// 懒惰删除的节点数
private BinaryNode<AnyType> min = null;
private BinaryNode<AnyType> max = null;
public myLazyBinarySearchTree() {
root = null;
this.deletedSize = 0;
this.theSize = 0;
}
public boolean isEmpty() {
return theSize - deletedSize == 0;
}
}
其中contains()比较简单,在递归查找中找到后再查看一下节点的flag变量即可;insert例程也一样,如果有是已存在元素,查看其flag变量是否=1。
下面是比较麻烦的remove()方法,实现方法和之前的LinkedList一样,先是递归删除,未找到直接return,找到了先将节点flag=1,改变theSize和deletedSize,然后比较theSize和deletedSize的值(懒惰删除的数目>=剩下的节点数目),进行标准删除。标准删除从root开始。
在标准删除中,如果节点不需要删除,直接检查其左右子树。如果检测到需要删除的节点,再检测①若只存在左子树或者右子树,则将子树拼接上来,重新检测该位置节点。②若是叶子节点,直接置Null删除。③若有左子树也右子树,则同理先将右子树找到一个最小的且flag=0的子树,这里又分为两种情况,若右子树不存在这样一个最小节点,则直接将该右子树置null,并重新检测该节点,若找到了则调整标志等。删除实现如下:
public void remove(AnyType x) {
remove(x, root);// 这里递归删除只能使用void在下面的remove的else中,t随着变化,而懒惰删除中,只有theSize<=deletedSize才变化,所以t时钟指向的root.造成错误。
}
private void remove(AnyType x, BinaryNode<AnyType> t) {
if (t == null) {
// 未找到
return;
}
int compareResult = x.compareTo(t.element);
if (compareResult < 0) {
remove(x, t.left);
} else if (compareResult > 0) {
remove(x, t.right);
} else {
// 找到了这个节点,标记删除
t.flag = 1;
this.theSize--;
this.deletedSize++;
checkDeletion();// 检测是否进行标准删除
System.out.println("root1:" + this.root.right);
}
}
private void checkDeletion() {
if (this.theSize <= this.deletedSize) {
this.root = doRemove(root);
// 更新节点
this.deletedSize = 0;
}
}
private BinaryNode<AnyType> doRemove(BinaryNode<AnyType> t) {// 删除树中所有flag=1的节点。使用递归
if (t == null) {
return null;
}
if (t.flag == 0) {// 不需要删除的节点,直接进入左右子树
//System.out.println("000");
t.left = doRemove(t.left);
t.right = doRemove(t.right);
} else {
if (t.left == null || t.right == null) {// 叶子节点或者只有一个子树
t = (t.left != null) ? t.left : t.right;
t = doRemove(t);// 检查拼接上来的子树
} else {// ②左右子树都存在
min = null;
findMin(t.right);
if (min == null) {// 右子树上所有节点都被懒惰删除,直接将t的右子树删除,并继续监测t
t.right = null;
t = doRemove(t);
} else {
t.element = min.element;
t.flag = 0;
min.flag = 1;
t = doRemove(t);
}
}
}
return t;
}
在删除需要查找右子树中最小数值节点且其flag=0,即是未被删除的最小节点。在这儿可以参考二叉树中序遍历的实现,可以利用栈后进先出的特点。为了简明易懂,我使用的是递归查找,并使用了一个前面定义的全局变量进行存储最小元。
private void findMin(BinaryNode<AnyType> t) {
if (t != null) {
findMin(t.left);
if (t.flag == 0 && min == null) {
min = t;
return;
}
findMin(t.right);
}
}
查找最大值方法类似。在删除的过程中,每次选择右子树最小元素来代替被删除的元素容易造成树的不平衡性(见P91),因而可以考虑进行随机删除,即随机算则右子树的最小元素或者左子树的最大元素,来消除树的偏向。
二叉查找树的懒惰删除(lazy deletion)的更多相关文章
- 【Weiss】【第03章】练习3.17:懒惰删除
[练习3.17] 不同于我们已经给出的删除方法,另一种是使用懒惰删除的方法. 为了删除一个元素,我们只标记上该元素被删除的信息(使用一个附加的位域). 表中被删除和非被删除的元素个数作为数据结构的一部 ...
- 二叉查找树(查找、插入、删除)——C语言
二叉查找树 二叉查找树(BST:Binary Search Tree)是一种特殊的二叉树,它改善了二叉树节点查找的效率.二叉查找树有以下性质: (1)若左子树不空,则左子树上所有节点的值均小于它的根节 ...
- 深入浅出数据结构C语言版(12)——平衡二叉查找树之AVL树
在上一篇博文中我们提到了,如果对普通二叉查找树进行随机的插入.删除,很可能导致树的严重不平衡 所以这一次,我们就来介绍一种最老的.可以实现左右子树"平衡效果"的树(或者说算法),即 ...
- JAVA数据结构--二叉查找树
二叉查找树定义 二叉查找树(英语:Binary Search Tree),也称二叉搜索树.有序二叉树(英语:ordered binary tree),排序二叉树(英语:sorted binary tr ...
- 数据结构--Avl树的创建,插入的递归版本和非递归版本,删除等操作
AVL树本质上还是一棵二叉搜索树,它的特点是: 1.本身首先是一棵二叉搜索树. 2.带有平衡条件:每个结点的左右子树的高度之差的绝对值最多为1(空树的高度为-1). 也就是说,AVL树,本质上 ...
- java:数据结构(四)二叉查找树以及树的三种遍历
@TOC 二叉树模型 二叉树是树的一种应用,一个节点可以有两个孩子:左孩子,右孩子,并且除了根节点以外每个节点都有一个父节点.当然这种简单的二叉树不能解决让树保持平衡状态,例如你一直往树的左边添加元素 ...
- 常见基本数据结构——树,二叉树,二叉查找树,AVL树
常见数据结构——树 处理大量的数据时,链表的线性时间太慢了,不宜使用.在树的数据结构中,其大部分的运行时间平均为O(logN).并且通过对树结构的修改,我们能够保证它的最坏情形下上述的时间界. 树的定 ...
- 二叉查找树的Java实现
为了克服对树结构编程的恐惧感,决心自己实现一遍二叉查找树,以便掌握关于树结构编程的一些技巧和方法.以下是基本思路: [1] 关于容器与封装.封装,是一种非常重要的系统设计思想:无论是面向过程的函数,还 ...
- 【查找结构 2】二叉查找树 [BST]
当所有的静态查找结构添加和删除一个数据的时候,整个结构都需要重建.这对于常常需要在查找过程中动态改变数据而言,是灾难性的.因此人们就必须去寻找高效的动态查找结构,我们在这讨论一个非常常用的动态查找树— ...
随机推荐
- C#强力粉碎文件代码分享,升级中用到
360的文件粉碎机还是很强大的,在我们客户端winform升级的时候,必须将有些文件进行强力删除后下载更新,如果删除失败很有可能整个 程序就无法更新到最新的版本,所以这里参考了网上的资料整理了一个文件 ...
- 内核移植和文件系统制作(4):UBIFS根文件系统制作总结
UBIFS文件系统简介: 无排序区块图像文件系统(UnsortedBlock Image File System, UBIFS)是用于固态硬盘存储设备上,并与LogFS相互竞争,作为JFFS2的后继文 ...
- gene框架文档 - 概述
欢迎使用Gene框架 最新版本:V1.2.2 开源地址:https://github.com/sasou/php-gene 作者:sasou 文档地址:http://php-gene.com/doc ...
- PHP系列之一traits的应用
Traits 在PHP中实现在方法的重复使用:Traits与Class相似,但是它能够在Class中使用自己的方法而不用继承: Traits在Class中优先于原Class中的方法,引用PHP Doc ...
- Javascript的封装
js的封装分为以下几种模式: 工厂模式,代码如下: <!doctype html><html lang="en"><head><meta ...
- Vue表单
gitHub地址: https://github.com/lily1010/vue_learn/tree/master/lesson11 一 vue表单 实在是太简单了,直接来个例子 <!DOC ...
- 破解 “PEDIY CrackMe 2007” 之 KeygenMe_1_by_boonz
系统 : Windows xp 程序 :KeygenMe_1_by_boonz 程序下载地址 :http://www.crackmes.de/users/boonz/keygenme_1_by_boo ...
- 深入底层逆向分析TDC‘s keygenme(手脱压缩壳)
系统 : Windows xp 程序 : TDC‘s keygenme 程序下载地址 :http://pan.baidu.com/s/1gdWyt6z 要求 : 脱壳 & 注册机编写 使用工具 ...
- Android 正则表达式
1.相关知识链接 正则表达式语法 https://msdn.microsoft.com/zh-cn/library/ae5bf541(v=vs.80).aspx 正则表达式入门教程 http: ...
- IOS编程思想
从今天起想走进IOS架构的大门,一直不屑于学习第三方框架,觉得框架也是一点点代码给垒起来的,只要掌握了代码就可以了,殊不知垒代码的过程才是最重要的,而这个过程又岂是一朝一夕就能达到完美境界的,达到完美 ...