0

0

java代码如何实现平衡二叉树的旋转操作 java代码平衡树维护的基础编写教程​

爱谁谁

爱谁谁

发布时间:2025-08-07 20:46:01

|

205人浏览过

|

来源于php中文网

原创

平衡二叉树的旋转操作是为了维持树的平衡性,防止其退化为链表,从而保证查找、插入、删除等操作的时间复杂度稳定在o(log n)。普通的二叉搜索树在插入有序数据时可能严重失衡,导致性能下降至o(n),而平衡二叉树通过旋转操作(如左旋、右旋)在节点失衡时调整结构,保持左右子树高度差不超过1。常见的平衡二叉树包括avl树、红黑树、b树和b+树:avl树严格保持平衡,查找效率高,但频繁旋转影响插入删除性能;红黑树牺牲部分平衡性以减少旋转次数,适合频繁修改的场景,广泛用于java集合类;b树和b+树为多路平衡树,适用于磁盘存储,其中b+树所有数据存于叶子节点,更支持高效范围查询,常用于数据库索引。测试平衡二叉树需从四个方面进行:1. 验证基本操作正确性,如插入、删除、查找;2. 检查平衡性,确保每次操作后所有节点的平衡因子绝对值不超过1;3. 进行性能测试,统计大量操作下的时间消耗是否符合o(log n)趋势;4. 覆盖边界条件,如空树、单节点、重复值、有序插入等情形。测试时应使用断言自动检测平衡性,结合可视化工具观察树形结构,并确保测试用例覆盖所有旋转情况和代码路径,以保障实现的正确性和鲁棒性。

java代码如何实现平衡二叉树的旋转操作 java代码平衡树维护的基础编写教程​

平衡二叉树的旋转操作是维持其平衡的关键。简单来说,当二叉树的某个节点左右子树高度差超过1时,就需要通过旋转来调整,避免树退化成链表,影响查找效率。

// 节点类
class Node {
    int data;
    Node left, right;
    int height; // 节点高度

    Node(int d) {
        data = d;
        height = 1; // 新节点高度为1
    }
}

// 平衡二叉树类
class AVLTree {
    Node root;

    // 获取节点高度
    int height(Node node) {
        if (node == null)
            return 0;
        return node.height;
    }

    // 更新节点高度
    void updateHeight(Node node) {
        node.height = Math.max(height(node.left), height(node.right)) + 1;
    }

    // 获取平衡因子(左子树高度 - 右子树高度)
    int getBalance(Node node) {
        if (node == null)
            return 0;
        return height(node.left) - height(node.right);
    }

    // 右旋操作
    Node rightRotate(Node y) {
        Node x = y.left;
        Node T2 = x.right;

        // 执行旋转
        x.right = y;
        y.left = T2;

        // 更新高度
        updateHeight(y);
        updateHeight(x);

        // 返回新的根节点
        return x;
    }

    // 左旋操作
    Node leftRotate(Node x) {
        Node y = x.right;
        Node T2 = y.left;

        // 执行旋转
        y.left = x;
        x.right = T2;

        // 更新高度
        updateHeight(x);
        updateHeight(y);

        // 返回新的根节点
        return y;
    }

    // 插入节点
    Node insert(Node node, int data) {
        // 1. 执行标准BST插入
        if (node == null)
            return (new Node(data));

        if (data < node.data)
            node.left = insert(node.left, data);
        else if (data > node.data)
            node.right = insert(node.right, data);
        else // 不允许重复值
            return node;

        // 2. 更新当前节点的高度
        updateHeight(node);

        // 3. 获取平衡因子
        int balance = getBalance(node);

        // 4. 如果节点不平衡,则有四种情况
        // 左左情况
        if (balance > 1 && data < node.left.data)
            return rightRotate(node);

        // 右右情况
        if (balance < -1 && data > node.right.data)
            return leftRotate(node);

        // 左右情况
        if (balance > 1 && data > node.left.data) {
            node.left = leftRotate(node.left);
            return rightRotate(node);
        }

        // 右左情况
        if (balance < -1 && data < node.right.data) {
            node.right = rightRotate(node.right);
            return leftRotate(node);
        }

        return node;
    }

    // 删除节点(简略,完整实现还需要考虑多种情况)
    Node deleteNode(Node root, int data) {
        if (root == null)
            return root;

        if (data < root.data)
            root.left = deleteNode(root.left, data);
        else if (data > root.data)
            root.right = deleteNode(root.right, data);
        else {
            // 节点是要删除的节点

            // 节点只有一个孩子或没有孩子
            if ((root.left == null) || (root.right == null)) {
                Node temp = null;
                if (temp == root.left)
                    temp = root.right;
                else
                    temp = root.left;

                // 没有孩子的情况
                if (temp == null) {
                    temp = root;
                    root = null;
                } else // 一个孩子的情况
                    root = temp; // 复制非空子节点
            } else {
                // 节点有两个孩子:获取中序后继(右子树中的最小节点)
                Node temp = minValueNode(root.right);

                // 将中序后继的值复制到该节点
                root.data = temp.data;

                // 删除中序后继
                root.right = deleteNode(root.right, temp.data);
            }
        }

        // 如果树只有一个节点,则返回
        if (root == null)
            return root;

        // 2. 更新当前节点的高度
        updateHeight(root);

        // 3. 获取平衡因子
        int balance = getBalance(root);

        // 如果节点不平衡,则有四种情况
        // 左左情况
        if (balance > 1 && getBalance(root.left) >= 0)
            return rightRotate(root);

        // 左右情况
        if (balance > 1 && getBalance(root.left) < 0) {
            root.left = leftRotate(root.left);
            return rightRotate(root);
        }

        // 右右情况
        if (balance < -1 && getBalance(root.right) <= 0)
            return leftRotate(root);

        // 右左情况
        if (balance < -1 && getBalance(root.right) > 0) {
            root.right = rightRotate(root.right);
            return leftRotate(root);
        }

        return root;
    }

    Node minValueNode(Node node) {
        Node current = node;

        /* 循环下降到最左边的叶子 */
        while (current.left != null)
            current = current.left;

        return current;
    }


    // 打印树(中序遍历)
    void preOrder(Node node) {
        if (node != null) {
            System.out.print(node.data + " ");
            preOrder(node.left);
            preOrder(node.right);
        }
    }
}


public class Main {
    public static void main(String[] args) {
        AVLTree tree = new AVLTree();
        tree.root = tree.insert(tree.root, 10);
        tree.root = tree.insert(tree.root, 20);
        tree.root = tree.insert(tree.root, 30);
        tree.root = tree.insert(tree.root, 40);
        tree.root = tree.insert(tree.root, 50);
        tree.root = tree.insert(tree.root, 25);

        System.out.println("Preorder traversal of constructed AVL tree is: ");
        tree.preOrder(tree.root);

        tree.root = tree.deleteNode(tree.root, 30);

        System.out.println("\nPreorder traversal after deletion of 30: ");
        tree.preOrder(tree.root);
    }
}

平衡二叉树的旋转操作,本质上是在维持二叉搜索树的性质(左子树小于根节点,右子树大于根节点)的前提下,调整树的结构,使其更加平衡。

为什么需要平衡二叉树?普通的二叉搜索树有什么问题?

普通的二叉搜索树在最坏情况下,可能退化成一个链表,导致查找、插入、删除等操作的时间复杂度从O(log n) 变成 O(n)。平衡二叉树通过旋转等操作,始终保持树的平衡,保证操作的时间复杂度维持在O(log n)级别。这对于需要频繁进行查找、插入、删除操作的应用场景非常重要。比如数据库索引,如果使用非平衡的二叉搜索树,性能会急剧下降。

立即学习Java免费学习笔记(深入)”;

除了AVL树,还有哪些常见的平衡二叉树?它们的区别是什么?

除了AVL树,常见的平衡二叉树还有红黑树、B树、B+树等。它们在平衡策略、实现复杂度、适用场景等方面有所不同:

  • 红黑树: 是一种近似平衡的二叉搜索树,通过对节点着色来维持平衡。相对于AVL树,红黑树的平衡性稍差,但插入、删除操作的平均性能更好,因为旋转次数更少。红黑树广泛应用于Java的TreeMap和TreeSet等数据结构中。

    Faceswap
    Faceswap

    免费开源的AI换脸工具

    下载
  • B树: 是一种多路搜索树,适合在磁盘等外部存储设备上使用。B树的特点是每个节点可以存储多个键值对,降低了树的高度,减少了磁盘I/O次数。

  • B+树: 是B树的变种,所有数据都存储在叶子节点上,非叶子节点只存储索引。B+树更适合范围查询,也更常用于数据库索引。

选择哪种平衡二叉树,取决于具体的应用场景和性能需求。如果插入、删除操作频繁,且对查找性能要求不是特别高,可以选择红黑树。如果数据存储在磁盘上,且需要支持范围查询,可以选择B+树。

如何测试平衡二叉树的正确性?有哪些需要注意的地方?

测试平衡二叉树的正确性,需要从多个方面进行验证:

  1. 基本操作测试: 验证插入、删除、查找等基本操作是否正确。可以构造一些典型的测试用例,比如插入有序序列、插入随机序列、删除根节点、删除叶子节点等。
  2. 平衡性测试: 验证树是否始终保持平衡。可以在每次插入或删除节点后,检查树的平衡因子是否超过允许的范围。
  3. 性能测试: 验证树的性能是否符合预期。可以生成大量随机数据,进行插入、删除、查找操作,并记录时间消耗。
  4. 边界条件测试: 验证树在边界条件下的行为是否正确。比如空树、只有一个节点的树、所有节点值都相同的树等。

在测试过程中,需要注意以下几点:

  • 使用断言: 在代码中使用断言,可以方便地检测错误。比如,可以在插入或删除节点后,断言树的平衡因子是否在允许范围内。
  • 可视化: 可以使用可视化工具,将树的结构显示出来,方便观察和调试。
  • 覆盖率: 确保测试用例覆盖了所有可能的代码路径。

总之,平衡二叉树的测试是一个复杂的过程,需要从多个方面进行验证,才能确保其正确性和可靠性。

相关文章

java速学教程(入门到精通)
java速学教程(入门到精通)

java怎么学习?java怎么入门?java在哪学?java怎么学才快?不用担心,这里为大家提供了java速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

537

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

25

2026.01.06

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

356

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2078

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

348

2023.08.31

MySQL恢复数据库
MySQL恢复数据库

MySQL恢复数据库的方法有使用物理备份恢复、使用逻辑备份恢复、使用二进制日志恢复和使用数据库复制进行恢复等。本专题为大家提供MySQL数据库相关的文章、下载、课程内容,供大家免费下载体验。

256

2023.09.05

vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

325

2023.10.09

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Git 教程
Git 教程

共21课时 | 3万人学习

Kotlin 教程
Kotlin 教程

共23课时 | 2.9万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号