数据结构

一、栈

先进后出

二、队列

先进先出

三、数组

查询快,增加修改慢

四、链表

查询慢,增加修改慢

五、二叉树

节点:

查找二叉树

二叉查找树的特点

  • 二叉查找树,又称二叉排序树或者二叉搜索树

  • 每一个节点上最多有两个子节点

  • 左子树上所有节点的值都小于根节点的值

  • 右子树上所有节点的值都大于根节点的值

二叉查找树添加节点规则

  • 小的存左边

  • 大的存右边

  • 一样的不存

数据结构(二叉树)遍历方式

  1. 前序遍历:当前节点,左子节点,右子结点
  2. 中序遍历:左子节点,当前节点,右子结点
  3. 后序遍历:左子节点,右子结点,当前节点
  4. 层序遍历:一层一层的去遍历

平衡二叉树

特点

  • 二叉树左右两个子树的高度差不超过1

  • 任意节点的左右两个子树都是一颗平衡二叉树

平衡二叉树旋转

  • 旋转触发时机

    • 当添加一个节点之后,该树不再是一颗平衡二叉树

  • 左旋

    • 确定支点:从添加的节点开始,不断的往父节点找不平衡的节点

    以不平衡的点作为支点
    将根节点的右侧往左拉
    原先的右子节点变成新的父节点,并把多余的左子节点出让,给已经降级的根节点当右子节点

右旋

  • 确定支点:从添加的节点开始,不断的往父节点找不平衡的节点

以不平衡的点作为支点
就是将根节点的左侧往右拉
原先的左子节点变成新的父节点,并把多余的右子节点出让,给已经降级的根节点当左子节点

平衡二叉树旋转的四种情况

左左

  • 左左: 当根节点左子树的左子树有节点插入,导致二叉树不平衡

  • 如何旋转: 直接对整体进行右旋即可

左右

  • 左右: 当根节点左子树的右子树有节点插入,导致二叉树不平衡

  • 如何旋转: 先在左子树对应的节点位置进行左旋,再对整体进行右旋

右右

  • 右右: 当根节点右子树的右子树有节点插入,导致二叉树不平衡

  • 如何旋转: 直接对整体进行左旋即可

右左

  • 右左:当根节点右子树的左子树有节点插入,导致二叉树不平衡

  • 如何旋转: 先在右子树对应的节点位置进行右旋,再对整体进行左旋

红黑树

红黑树的特点

红黑树的增删改查性能都很好

  • 平衡二叉B树

  • 每一个节点可以是红或者黑

  • 红黑树不是高度平衡的,它的平衡是通过"自己的红黑规则"进行实现的

节点

红黑规则

  1. 每一个节点是红色的,或者是黑色的
  2. 根节点必须是黑色
  3. 叶节点是黑色的
  4. 两个红色节点不能相连
  5. 任意节点到所有后代叶节点的简单路径上,黑色节点数量相同;

红黑树结构图

红黑树添加节点的默认颜色

  • 添加节点时,默认为红色,效率高

红黑树添加节点后如何保持红黑规则

  • 根节点位置

    • 直接变为黑色

  • 非根节点位置

    • 父节点为黑色

      • 不需要任何操作,默认红色即可

    • 父节点为红色

      • 叔叔节点为红色

        1. 将"父节点"设为黑色,将"叔叔节点"设为黑色

        2. 将"祖父节点"设为红色

        3. 如果"祖父节点"为根节点,则将根节点再次变成黑色

      • 叔叔节点为黑色

        1. 将"父节点"设为黑色

        2. 将"祖父节点"设为红色

        3. 以"祖父节点"为支点进行旋转

实现代码

public class RBTree {
    class Node {
        int val, color;
        Node left, right;
    }
//    使用NIL节点来充当叶节点
    Node NIL;
    Node root;

    public RBTree() {
        NIL = new Node();
        NIL.val = -1;
        NIL.color = 1;
        NIL.left = NIL.right = NIL;
        root = NIL;
    }
    // 创建节点
    private Node getNewNode(int val) {
        Node p = new Node();
        p.val = val;
        p.color = 0;
        p.left = p.right = NIL;
        return p;
    }
    //判断有没有红色孩子
    private boolean has_red_child(Node tree) {
        return tree.left.color == 0 || tree.right.color == 0;
    }
    //左旋
    private Node left_rotate(Node tree) {
        Node temp = tree.right;
        tree.right = temp.left;
        temp.left = tree;
        return temp;
    }
    //右旋
    private Node right_rotate(Node tree) {
        Node temp = tree.left;
        tree.left = temp.right;
        temp.right = tree;
        return temp;
    }

    //寻找前驱
    private Node preNode(Node tree) {
        Node p = tree.left;
        while (p.right != null) {
            p = p.right;
        }
        return p;
    }
    //删除
    public void erase(int val) {
        root = erase(root, val);
    }

    private Node erase(Node tree, int val) {
        tree = __erase(tree, val);
        tree.color = 1;
        return tree;
    }

    private Node __erase(Node tree, int val) {
        if (tree == NIL) return tree;
        if (val < tree.val) {
            tree.left = __erase(tree.left, val);
        } else if (val > tree.val) {
            tree.right = __erase(tree.right, val);
        } else {
            if (tree.left == NIL || tree.right == NIL) {
                Node temp = tree.left == NIL ? tree.right : tree.left;
                temp.color += tree.color;
                tree = temp;
                return tree;
            } else {
                Node temp = preNode(tree);
                tree.val = temp.val;
                tree.left = __erase(tree.left, temp.val);
            }
        }
        return erase_maintion(tree);
    }
    //删除调整
    private Node erase_maintion(Node tree) {
        if (tree.left.color != 2 && tree.right.color != 2) return tree;
//        兄弟为红,旋转树,新根节点转为黑,原根节点转为红
        if (has_red_child(tree)) {
            int flag = 0;
            tree.color = 0;
            if (tree.left.color == 0) {
                tree = right_rotate(tree);
                flag = 1;
            } else {
                tree = left_rotate(tree);
            }
            tree.color = 1;
            if (flag == 1) tree.right = erase_maintion(tree.right);
            else tree.left = erase_maintion(tree.left);
            return tree;
        }
//        兄弟为黑色并且没有红色子节点,子节点减黑,根节点加黑
        if (tree.left.color == 1 && !has_red_child(tree.left)
                || tree.right.color == 1 && !has_red_child(tree.right)) {
            tree.color += 1;
            tree.left.color -= 1;
            tree.right.color -= 1;
            return tree;
        }
//        兄弟节点为黑并且有红色子节点
//            |-- 左子树为黑色
//              |-- 左子树的右子树为红色且左子树节点为黑 LR
//                  |-- 子树小左旋,新节点转黑,原节点转红,进入LL形态
//              |-- 左子树的左子树为红色 LL
//                  |-- 整树右旋,新节点改为原根节点的颜色,原根节点已经新叔叔节点转为黑色
//            |-- 右子树为黑色
//              |-- 右子树的左子树为红色且右子树节点为黑 RL
//                  |-- 子树小右旋,新节点转黑,原节点转红,进入RR形态
//              |-- 右子树的右子树为红色 RR
//                  |-- 整树左旋,新节点改为原根节点的颜色,原根节点已经新叔叔节点转为黑色
        if (tree.left.color == 1) {
            tree.right.color = 1;
            if (tree.left.left.color != 0) {
                tree.left.color = 0;
                tree.left = left_rotate(tree.left);
                tree.left.color = 1;
            }
            tree.left.color = tree.color;
            tree = right_rotate(tree);
        } else {
            tree.left.color = 1;
            if (tree.right.right.color != 0) {
                tree.right.color = 0;
                tree.right = right_rotate(tree.right);
                tree.right.color = 1;
            }
            tree.right.color = tree.color;
            tree = left_rotate(tree);
        }
        tree.left.color = 1;
        tree.right.color = 1;
        return tree;
    }
    //添加
    public void insert(int val) {
        root = insert(root, val);
    }

    private Node insert(Node tree, int val) {
        tree = __insert(tree, val);
        tree.color = 1;
        return tree;
    }

    private Node __insert(Node tree, int val) {
        if (tree == NIL) {
            return getNewNode(val);
        }
        if (val < tree.val) {
            tree.left = __insert(tree.left, val);
        } else if (val > tree.val) {
            tree.right = __insert(tree.right, val);
        }
        return insert_maintain(tree);
    }
    //添加调整
    private Node insert_maintain(Node tree) {
        if (!has_red_child(tree)) return tree;
        //节点双红
        if (tree.left.color == 0 && tree.right.color == 0) {
            if (!has_red_child(tree.left) && !has_red_child(tree.right)) return tree;
            tree.color = 0;
            tree.left.color = tree.right.color = 1;
            return tree;
        }
        if (tree.left.color == 0 && !has_red_child(tree.left)) return tree;
        if (tree.right.color == 0 && !has_red_child(tree.right)) return tree;
        // 左子树失衡
        if (tree.left.color == 0) {
            if (tree.left.right.color == 0) {
                tree.left = left_rotate(tree.left);
            }
            tree = right_rotate(tree);
        } else {
            if (tree.right.left.color == 0) {
                tree.right = right_rotate(tree.right);
            }
            tree = left_rotate(tree);
        }
        tree.color = 0;
        tree.left.color = tree.right.color = 1;
        return tree;
    }
    //打印输出
    public void preorder() {
        preorder(root, root.val, 0);
    }

    private void preorder(Node tree, int val, int flag) {
        if (tree == NIL) return;
        if (flag == 0) {
            System.out.printf("%d is root, color is %s\n", val, tree.color == 0 ? "red" : "black");
        } else {
            System.out.printf("%d is %d's %s child, color is %s\n"
                    , tree.val, val, flag == 1 ? "right" : "left", tree.color == 0 ? "red" : "black");
        }
        preorder(tree.left, tree.val, -1);
        preorder(tree.right, tree.val, 1);

    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/491931.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

QQ屠龙者终成为恶龙!工作与生活的平衡:挑战与解决方案——早读(逆天打工人爬取热门微信文章解读)

屠龙者终成为恶龙 引言Python 代码第一篇 洞见 看了上海临时工睡桥洞的现场&#xff0c;我不敢让老板知道上班苦第二篇 人民日报 来了&#xff01;新闻早班车要闻社会政策 结尾 昔日斩棘披荆的屠龙勇士 历经岁月涤荡 竟在不知不觉间铸就了自己心中的铁壁铜墙 终有一日 赫然发现…

【智能计算模拟:DFT+MD+ML深度融合及科研实践应用】

第一性原理、分子动力学与机器学习三者的交汇融合已在相关研究领域展现强劲的研究热潮。借助第一性原理计算揭示材料内在的量子特性&#xff0c;并结合分子动力学模拟探究材料在实际环境下的动态行为&#xff1b;运用机器学习算法与上述方法结合&#xff0c;开发高性能预测模型…

智能医疗-方案优势

实时更新&#xff0c;信息展示更便利 电子标签床头信息卡可设计特定的信息模板展示病患信息&#xff0c;并可根据治疗进展实时更新&#xff0c;病患通过床头电子标签即可了解病情信息。 —签多能&#xff0c;医护管理更高效 电子墨水屏技术改进了传统院内展示内容单一、更换成…

edga 0x800704cf错误码(同步失败)

edga 0x800704cf错误码(同步失败) 执行此操作需要 Internet。 你似乎没有连接到 Internet。请检查你的连接&#xff0c;然后再试一次。 0x800704cf 发送反馈 取消windows键R&#xff0c; 输入services.msc 进入本地服务管理&#xff0c; 重启 Microsoft Account Sign-in Assi…

【探索Linux】—— 强大的命令行工具 P.30(序列化和反序列化 | JSON序列化库 [ C++ ] )

阅读导航 引言一、序列化反序列化概念二、各种语言序列化反序列化常用库三、C的JSON序列化库1. RapidJSON库&#xff08;1&#xff09;头文件&#xff08;2&#xff09;相关函数&#xff08;3&#xff09;参数&#xff08;4&#xff09;特点 &#xff08;5&#xff09;使用示例…

零拷贝技术、常见实现方案、Kafka中的零拷贝技术的使用、Kafka为什么这么快

目录 1. 普通拷贝 2. 数据拷贝基础过程 2.1 仅CPU方式 2.2 CPU&DMA方式 3.普通模式数据交互 4. 零拷贝技术 4.1 出现原因 4.2 解决思路 4.2.1 mmap方式 4.2.2 sendfile方式 4.2.3 sendfileDMA收集 4.2.4 splice方式 5. Kafka中使用到的零拷贝技术 参考链接 本…

windows下powershell与linux下bash美化教程(使用starship)

starship美化教程 Win11 Powershell 安装 在命令行使用下面命令安装 # 安装starship winget install starship将以下内容添加到 Microsoft.PowerShell_profile.ps1&#xff0c;可以在 PowerShell 通过 $PROFILE 变量来查询文件的位置 Invoke-Expression (&starship i…

gitee多用户配置

一、引言 在工作的时候我们有时候会自己创建项目Demo来实现一些功能&#xff0c;但是又不想把自己的Demo代码放到公司的仓库代码平台&#xff08;gitee&#xff09;中管理&#xff0c;于是就是想自己放到自己的Gitee中管理&#xff0c;于是就需要配置Git多用户。 本文将配置分别…

【4月】组队打卡《山有木兮Python数据分析极简入门》

活动名称 CDA Club 第2期《山有木兮Python数据分析极简入门》组队打卡 活动介绍 本次打卡活动由CDA俱乐部旗下学术部主办。目的是通过数据分析科普内容&#xff0c;为数据分析爱好者提供学习和交流的机会。方便大家利用碎片化时间在线学习&#xff0c;以组队打卡的形式提升学…

Kindling the Darkness:A Practical Low-light Image Enhancer

Abstract 在弱光条件下拍摄的图像通常会出现&#xff08;部分&#xff09;可见度较差的情况。,除了令人不满意的照明之外&#xff0c;多种类型的退化也隐藏在黑暗中&#xff0c;例如由于相机质量有限而导致的噪点和颜色失真。,换句话说&#xff0c;仅仅调高黑暗区域的亮度将不…

Altair(澳汰尔) Radioss® 评估和优化动态载荷下的高度非线性问题

Altair&#xff08;澳汰尔&#xff09; Radioss 评估和优化动态载荷下的高度非线性问题 Radioss 是一款超前的分析解决方案&#xff0c;可评估和优化动态载荷下的高度非线性问题。它广泛应用于全球各行各业&#xff0c;能有效提高复杂设计的耐撞性、安全性和可制造性。 30 多…

iOS - Runtime - Class的结构

文章目录 iOS - Runtime - Class的结构前言1. Class的结构1.1 Class的结构1.1.1 objc_class1.1.2 class_rw_t1.1.3 class_ro_t 1.2 class_rw_t和class_ro_t的区别1.3 class_rw_t和class_ro_t的关系1.3.1 分析关系1.3.2 原因 1.4 method_t1.4.1 Type Encoding1.4.2 types iOS - …

00000基础搭建vue+flask前后端分离项目

我完全是参考的这个vue3flask前后端分离环境速建_flask vue3-CSDN博客 安装了node_js&#xff08;添加了环境变量&#xff09; 环境变量 把原来的镜像源换成了淘宝镜像源 npm config set registry https://registry.npmmirror.com/ 查看版本证明安装成功 npm - v 安装npm i…

web自动化测试系列-selenium的运行原理和常用方法介绍(二)

目录 1.selenium的运行原理 2.常用方法介绍 接上文 &#xff1a;web自动化测试系列-selenium的安装和运行(一)-CSDN博客 在上文中我们编写了一段简单的代码 &#xff0c;可以驱动浏览器访问百度并搜索关键字 。这里我们再把这段代码再拿来加以说明 。 # 1. 导包 from selen…

gitee拉取与推送

&#x1f331;博客主页&#xff1a;青竹雾色间 &#x1f618;博客制作不易欢迎各位&#x1f44d;点赞⭐收藏➕关注 目录 一&#xff0c;从本地推送项目到gitee1.首先我们在gitee上创建一个仓库2.clone远程仓库到本地3.git的三板斧3.1. add - 将代码添加到本地仓库3.2. commit …

入行AI写作第一个月收入2万+复盘分享

AI写作作为一种新兴的创作方式&#xff0c;正逐渐改变着内容产业的生态。在这个领域中&#xff0c;许多人通过自己的努力和智慧&#xff0c;实现了快速的成长和收入的增长。本文将从技术学习与掌握、实践与应用、内容创作与优化、持续学习与创新、总结与复盘这五个方面&#xf…

Python时间序列异常检测库之adtk使用详解

概要 ADTK(Anomaly Detection Toolkit)是一个用于时间序列异常检测的Python库。它提供了一系列工具来识别时间序列数据中的异常模式,适用于多种监测和预测任务。 ADTK简介 ADTK专注于时间序列异常检测,支持多种异常检测算法,包括统计学方法、机器学习模型以及基于规则的…

配置Web运行环境与第一个网页

安装与配置Web环境: 如下使用了VSC作为web的运行环境。 下面是VSC的官网点击进入:Download Visual Studio Code - Mac, Linux, Windowshttps://code.visualstudio.com/download 1.下载 进入官网后可以看到有windows&#xff0c;linux&#xff0c;还有苹果按照自己的系统下载&…

Halcon深度学习项目实战系统教程

Halcon在机器视觉中的价值主要体现在提供高效、可扩展、灵活的机器视觉解决方案,帮助用户解决各种复杂的机器视觉问题,提高生产效率和产品质量。 缩短产品上市时间 Halcon的灵活架构使其能够快速开发出任何类型的机器视觉应用。其全球通用的集成开发环境(HDevelop)有助于降…

CDLKF不锈钢浸入式多级离心泵

一、构造特点与工作原理 CDLKF型不锈钢浸入式多级离心泵采用了先进的设计理念&#xff0c;其结构紧凑&#xff0c;占地面积小。泵体采用优质不锈钢材料&#xff0c;不仅保证了良好的抗腐蚀性和耐磨性&#xff0c;同时也确保了泵的长期稳定运行。 该泵的核心部件是多级离心叶轮&…