文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

JDK中如何删除元素

2023-06-20 13:49

关注

这篇文章将为大家详细讲解有关JDK中如何删除元素,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。

删除元素

删除元素本身比较简单,就是采用二叉树的删除规则。

public V remove(Object key) {    // 获取节点    Entry<K,V> p = getEntry(key);    if (p == null)        return null;    V oldValue = p.value;    // 删除节点    deleteEntry(p);    // 返回删除的value    return oldValue;}private void deleteEntry(Entry<K,V> p) {    // 修改次数加1    modCount++;    // 元素个数减1    size--;    if (p.left != null && p.right != null) {        // 如果当前节点既有左子节点,又有右子节点        // 取其右子树中最小的节点        Entry<K,V> s = successor(p);        // 用右子树中最小节点的值替换当前节点的值        p.key = s.key;        p.value = s.value;        // 把右子树中最小节点设为当前节点        p = s;        // 这种情况实际上并没有删除p节点,而是把p节点的值改了,实际删除的是p的后继节点    }    // 如果原来的当前节点(p)有2个子节点,则当前节点已经变成原来p的右子树中的最小节点了,也就是说其没有左子节点了    // 到这一步,p肯定只有一个子节点了    // 如果当前节点有子节点,则用子节点替换当前节点    Entry<K,V> replacement = (p.left != null ? p.left : p.right);    if (replacement != null) {        // 把替换节点直接放到当前节点的位置上(相当于删除了p,并把替换节点移动过来了)        replacement.parent = p.parent;        if (p.parent == null)            root = replacement;        else if (p == p.parent.left)            p.parent.left  = replacement;        else            p.parent.right = replacement;        // 将p的各项属性都设为空        p.left = p.right = p.parent = null;        // 如果p是黑节点,则需要再平衡        if (p.color == BLACK)            fixAfterDeletion(replacement);    } else if (p.parent == null) {        // 如果当前节点就是根节点,则直接将根节点设为空即可        root = null;    } else {        // 如果当前节点没有子节点且其为黑节点,则把自己当作虚拟的替换节点进行再平衡        if (p.color == BLACK)            fixAfterDeletion(p);        // 平衡完成后删除当前节点(与父节点断绝关系)        if (p.parent != null) {            if (p == p.parent.left)                p.parent.left = null;            else if (p == p.parent.right)                p.parent.right = null;            p.parent = null;        }    }}

删除再平衡

经过上面的处理,真正删除的肯定是黑色节点才会进入到再平衡阶段。

因为删除的是黑色节点,导致整颗树不平衡了,所以这里我们假设把删除的黑色赋予当前节点,这样当前节点除了它自已的颜色还多了一个黑色,那么:

(1)如果当前节点是根节点,则直接涂黑即可,不需要再平衡;

(2)如果当前节点是红+黑节点,则直接涂黑即可,不需要平衡;

(3)如果当前节点是黑+黑节点,则我们只要通过旋转把这个多出来的黑色不断的向上传递到一个红色节点即可,这又可能会出现以下四种情况:

假设当前节点为父节点的左子节点

情况策略
1)x是黑+黑节点,x的兄弟是红节点(1)将兄弟节点设为黑色; (2)将父节点设为红色; (3)以父节点为支点进行左旋; (4)重新设置x的兄弟节点,进入下一步;
2)x是黑+黑节点,x的兄弟是黑节点,且兄弟节点的两个子节点都是黑色(1)将兄弟节点设置为红色; (2)将x的父节点作为新的当前节点,进入下一次循环;
3)x是黑+黑节点,x的兄弟是黑节点,且兄弟节点的右子节点为黑色,左子节点为红色(1)将兄弟节点的左子节点设为黑色; (2)将兄弟节点设为红色; (3)以兄弟节点为支点进行右旋; (4)重新设置x的兄弟节点,进入下一步;
3)x是黑+黑节点,x的兄弟是黑节点,且兄弟节点的右子节点为红色,左子节点任意颜色(1)将兄弟节点的颜色设为父节点的颜色; (2)将父节点设为黑色; (3)将兄弟节点的右子节点设为黑色; (4)以父节点为支点进行左旋; (5)将root作为新的当前节点(退出循环);

假设当前节点为父节点的右子节点,正好反过来

情况策略
1)x是黑+黑节点,x的兄弟是红节点(1)将兄弟节点设为黑色; (2)将父节点设为红色; (3)以父节点为支点进行右旋; (4)重新设置x的兄弟节点,进入下一步;
2)x是黑+黑节点,x的兄弟是黑节点,且兄弟节点的两个子节点都是黑色(1)将兄弟节点设置为红色; (2)将x的父节点作为新的当前节点,进入下一次循环;
3)x是黑+黑节点,x的兄弟是黑节点,且兄弟节点的左子节点为黑色,右子节点为红色(1)将兄弟节点的右子节点设为黑色; (2)将兄弟节点设为红色; (3)以兄弟节点为支点进行左旋; (4)重新设置x的兄弟节点,进入下一步;
3)x是黑+黑节点,x的兄弟是黑节点,且兄弟节点的左子节点为红色,右子节点任意颜色(1)将兄弟节点的颜色设为父节点的颜色; (2)将父节点设为黑色; (3)将兄弟节点的左子节点设为黑色; (4)以父节点为支点进行右旋; (5)将root作为新的当前节点(退出循环);

让我们来看看TreeMap中的实现:

private void fixAfterDeletion(Entry<K,V> x) {    // 只有当前节点不是根节点且当前节点是黑色时才进入循环    while (x != root && colorOf(x) == BLACK) {        if (x == leftOf(parentOf(x))) {            // 如果当前节点是其父节点的左子节点            // sib是当前节点的兄弟节点            Entry<K,V> sib = rightOf(parentOf(x));            // 情况1)如果兄弟节点是红色            if (colorOf(sib) == RED) {                // (1)将兄弟节点设为黑色                setColor(sib, BLACK);                // (2)将父节点设为红色                setColor(parentOf(x), RED);                // (3)以父节点为支点进行左旋                rotateLeft(parentOf(x));                // (4)重新设置x的兄弟节点,进入下一步                sib = rightOf(parentOf(x));            }            if (colorOf(leftOf(sib))  == BLACK &&                    colorOf(rightOf(sib)) == BLACK) {                // 情况2)如果兄弟节点的两个子节点都是黑色                // (1)将兄弟节点设置为红色                setColor(sib, RED);                // (2)将x的父节点作为新的当前节点,进入下一次循环                x = parentOf(x);            } else {                if (colorOf(rightOf(sib)) == BLACK) {                    // 情况3)如果兄弟节点的右子节点为黑色                    // (1)将兄弟节点的左子节点设为黑色                    setColor(leftOf(sib), BLACK);                    // (2)将兄弟节点设为红色                    setColor(sib, RED);                    // (3)以兄弟节点为支点进行右旋                    rotateRight(sib);                    // (4)重新设置x的兄弟节点                    sib = rightOf(parentOf(x));                }                // 情况4)                // (1)将兄弟节点的颜色设为父节点的颜色                setColor(sib, colorOf(parentOf(x)));                // (2)将父节点设为黑色                setColor(parentOf(x), BLACK);                // (3)将兄弟节点的右子节点设为黑色                setColor(rightOf(sib), BLACK);                // (4)以父节点为支点进行左旋                rotateLeft(parentOf(x));                // (5)将root作为新的当前节点(退出循环)                x = root;            }        } else { // symmetric            // 如果当前节点是其父节点的右子节点            // sib是当前节点的兄弟节点            Entry<K,V> sib = leftOf(parentOf(x));            // 情况1)如果兄弟节点是红色            if (colorOf(sib) == RED) {                // (1)将兄弟节点设为黑色                setColor(sib, BLACK);                // (2)将父节点设为红色                setColor(parentOf(x), RED);                // (3)以父节点为支点进行右旋                rotateRight(parentOf(x));                // (4)重新设置x的兄弟节点                sib = leftOf(parentOf(x));            }            if (colorOf(rightOf(sib)) == BLACK &&                    colorOf(leftOf(sib)) == BLACK) {                // 情况2)如果兄弟节点的两个子节点都是黑色                // (1)将兄弟节点设置为红色                setColor(sib, RED);                // (2)将x的父节点作为新的当前节点,进入下一次循环                x = parentOf(x);            } else {                if (colorOf(leftOf(sib)) == BLACK) {                    // 情况3)如果兄弟节点的左子节点为黑色                    // (1)将兄弟节点的右子节点设为黑色                    setColor(rightOf(sib), BLACK);                    // (2)将兄弟节点设为红色                    setColor(sib, RED);                    // (3)以兄弟节点为支点进行左旋                    rotateLeft(sib);                    // (4)重新设置x的兄弟节点                    sib = leftOf(parentOf(x));                }                // 情况4)                // (1)将兄弟节点的颜色设为父节点的颜色                setColor(sib, colorOf(parentOf(x)));                // (2)将父节点设为黑色                setColor(parentOf(x), BLACK);                // (3)将兄弟节点的左子节点设为黑色                setColor(leftOf(sib), BLACK);                // (4)以父节点为支点进行右旋                rotateRight(parentOf(x));                // (5)将root作为新的当前节点(退出循环)                x = root;            }        }    }    // 退出条件为多出来的黑色向上传递到了根节点或者红节点    // 则将x设为黑色即可满足红黑树规则    setColor(x, BLACK);}

删除元素举例

假设我们有下面这样一颗红黑树。

JDK中如何删除元素

我们删除6号元素,则从右子树中找到了最小元素7,7又没有子节点了,所以把7作为当前节点进行再平衡。

我们看到7是黑节点,且其兄弟为黑节点,且其兄弟的两个子节点都是红色,满足情况4),平衡之后如下图所示。

JDK中如何删除元素

我们再删除7号元素,则从右子树中找到了最小元素8,8有子节点且为黑色,所以8的子节点9是替代节点,以9为当前节点进行再平衡。

我们发现9是红节点,则直接把它涂成黑色即满足了红黑树的特性,不需要再过多的平衡了。

JDK中如何删除元素

这次我们来个狠的,把根节点删除,从右子树中找到了最小的元素5,5没有子节点,所以把5作为当前节点进行再平衡。

我们看到5是黑节点,且其兄弟为红色,符合情况1),平衡之后如下图所示,然后进入情况2)。

JDK中如何删除元素

对情况2)进行再平衡后如下图所示。

JDK中如何删除元素

然后进入下一次循环,发现不符合循环条件了,直接把x涂为黑色即可,退出这个方法之后会把旧x删除掉(见deleteEntry()方法),最后的结果就是下面这样。

JDK中如何删除元素

二叉树的遍历

我们知道二叉查找树的遍历有前序遍历、中序遍历、后序遍历。

(1)前序遍历,先遍历我,再遍历我的左子节点,最后遍历我的右子节点;

(2)中序遍历,先遍历我的左子节点,再遍历我,最后遍历我的右子节点;

(3)后序遍历,先遍历我的左子节点,再遍历我的右子节点,最后遍历我;

这里的前中后都是以“我”的顺序为准的,我在前就是前序遍历,我在中就是中序遍历,我在后就是后序遍历。

下面让我们看看经典的中序遍历是怎么实现的:

public class TreeMapTest {    public static void main(String[] args) {        // 构建一颗10个元素的树        TreeNode<Integer> node = new TreeNode<>(1, null).insert(2)                .insert(6).insert(3).insert(5).insert(9)                .insert(7).insert(8).insert(4).insert(10);        // 中序遍历,打印结果为1到10的顺序        node.root().inOrderTraverse();    }}class TreeNode<T extends Comparable<T>> {    T value;    TreeNode<T> parent;    TreeNode<T> left, right;    public TreeNode(T value, TreeNode<T> parent) {        this.value = value;        this.parent = parent;    }        TreeNode<T> root() {        TreeNode<T> cur = this;        while (cur.parent != null) {            cur = cur.parent;        }        return cur;    }        void inOrderTraverse() {        if(this.left != null) this.left.inOrderTraverse();        System.out.println(this.value);        if(this.right != null) this.right.inOrderTraverse();    }        TreeNode<T> insert(T value) {        // 先找根元素        TreeNode<T> cur = root();        TreeNode<T> p;        int dir;        // 寻找元素应该插入的位置        do {            p = cur;            if ((dir=value.compareTo(p.value)) < 0) {                cur = cur.left;            } else {                cur = cur.right;            }        } while (cur != null);        // 把元素放到找到的位置        if (dir < 0) {            p.left = new TreeNode<>(value, p);            return p.left;        } else {            p.right = new TreeNode<>(value, p);            return p.right;        }    }}

TreeMap的遍历

从上面二叉树的遍历我们很明显地看到,它是通过递归的方式实现的,但是递归会占用额外的空间,直接到线程栈整个释放掉才会把方法中申请的变量销毁掉,所以当元素特别多的时候是一件很危险的事。

(上面的例子中,没有申请额外的空间,如果有声明变量,则可以理解为直到方法完成才会销毁变量)

那么,有没有什么方法不用递归呢?

让我们来看看java中的实现:

@Overridepublic void forEach(BiConsumer<? super K, ? super V> action) {    Objects.requireNonNull(action);    // 遍历前的修改次数    int expectedModCount = modCount;    // 执行遍历,先获取第一个元素的位置,再循环遍历后继节点    for (Entry<K, V> e = getFirstEntry(); e != null; e = successor(e)) {        // 执行动作        action.accept(e.key, e.value);        // 如果发现修改次数变了,则抛出异常        if (expectedModCount != modCount) {            throw new ConcurrentModificationException();        }    }}

是不是很简单?!

(1)寻找第一个节点;

从根节点开始找最左边的节点,即最小的元素。

final Entry<K,V> getFirstEntry() {        Entry<K,V> p = root;        // 从根节点开始找最左边的节点,即最小的元素        if (p != null)            while (p.left != null)                p = p.left;        return p;    }

(2)循环遍历后继节点;

寻找后继节点这个方法我们在删除元素的时候也用到过,当时的场景是有右子树,则从其右子树中寻找最小的节点。

static <K,V> TreeMap.Entry<K,V> successor(Entry<K,V> t) {    if (t == null)        // 如果当前节点为空,返回空        return null;    else if (t.right != null) {        // 如果当前节点有右子树,取右子树中最小的节点        Entry<K,V> p = t.right;        while (p.left != null)            p = p.left;        return p;    } else {        // 如果当前节点没有右子树        // 如果当前节点是父节点的左子节点,直接返回父节点        // 如果当前节点是父节点的右子节点,一直往上找,直到找到一个祖先节点是其父节点的左子节点为止,返回这个祖先节点的父节点        Entry<K,V> p = t.parent;        Entry<K,V> ch = t;        while (p != null && ch == p.right) {            ch = p;            p = p.parent;        }        return p;    }}

让我们一起来分析下这种方式的时间复杂度吧。

首先,寻找第一个元素,因为红黑树是接近平衡的二叉树,所以找最小的节点,相当于是从顶到底了,时间复杂度为O(log n);

其次,寻找后继节点,因为红黑树插入元素的时候会自动平衡,最坏的情况就是寻找右子树中最小的节点,时间复杂度为O(log k),k为右子树元素个数;

最后,需要遍历所有元素,时间复杂度为O(n);

所以,总的时间复杂度为 O(log n) + O(n * log k) ≈ O(n)。

虽然遍历红黑树的时间复杂度是O(n),但是它实际是要比跳表要慢一点的。

关于“JDK中如何删除元素”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯