news 2026/5/1 8:42:17

二叉树操作全解析:从构建到删除

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
二叉树操作全解析:从构建到删除

一、二叉树基础与节点定义

二叉树是计算机科学中最基本、最重要的数据结构之一,它是一种每个节点最多有两个子节点的树形结构。这两个子节点通常被称为左子节点右子节点。二叉树在算法设计、数据库索引、文件系统等众多领域都有广泛应用。

二叉树节点的Java实现

在Java中,我们首先需要定义二叉树节点的基本结构。下面是节点类的实现:

// TreeNode.java - 二叉树节点定义 package com.qcby; public class TreeNode { public TreeNode lChild; // 左子节点引用 public TreeNode rChild; // 右子节点引用 public Integer data; // 节点存储的数据 // 构造函数,初始化节点数据 public TreeNode(Integer data){ this.data = data; } }

这个简洁的TreeNode类定义了三个核心属性:

  • data:存储节点的整数值

  • lChild:指向左子节点的引用

  • rChild:指向右子节点的引用

二、二叉搜索树的构建

二叉搜索树(Binary Search Tree, BST)是一种特殊的二叉树,它满足以下性质:

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

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

  • 左右子树也都是二叉搜索树

下面是二叉搜索树的构建方法实现:

// BinaryTree.java - 二叉搜索树的创建方法 public void create(Integer value){ // 1.创建新节点 TreeNode newNode = new TreeNode(value); if(root==null){ root = newNode; return; } TreeNode curNode = root; while(true){ if(curNode.data < newNode.data){ if(curNode.rChild == null){ curNode.rChild = newNode; return; } curNode = curNode.rChild; }else{ if(curNode.lChild == null){ curNode.lChild = newNode; return; } curNode = curNode.lChild; } } }

插入算法解析:

  1. 创建新节点:根据传入的值创建新的TreeNode对象

  2. 检查空树:如果根节点为null,直接将新节点设为根节点

  3. 查找插入位置

    • 从根节点开始遍历

    • 如果新节点值大于当前节点值,向右子树移动

    • 如果新节点值小于等于当前节点值,向左子树移动

  4. 插入新节点:找到合适的位置(空位置)后,将新节点插入

时间复杂度分析

  • 平均情况:O(log n),其中n是节点数

  • 最坏情况:O(n),当树退化成链表时

三、二叉树的遍历方法

遍历是访问二叉树中所有节点的过程,确保每个节点仅被访问一次。以下是四种基本遍历方法的实现:

1. 前序遍历(Pre-order Traversal)

访问顺序:根节点 → 左子树 → 右子树

// BinaryTree.java - 前序遍历实现 void beforeOrder(TreeNode root){ if(root == null){ return; } System.out.println(root.data); beforeOrder(root.lChild); beforeOrder(root.rChild); }

算法特点

  • 先访问根节点,再递归遍历左子树,最后递归遍历右子树

  • 适用于需要先处理父节点再处理子节点的场景

应用场景

  • 复制二叉树结构

  • 计算前缀表达式

  • 序列化二叉树

2. 中序遍历(In-order Traversal)

访问顺序:左子树 → 根节点 → 右子树

// BinaryTree.java - 中序遍历实现 void inOrder(TreeNode root){ if(root == null){ return; } inOrder(root.lChild); System.out.println(root.data); inOrder(root.rChild); }

重要特性:对二叉搜索树进行中序遍历,会得到一个有序序列

应用场景

  • 输出有序数据

  • 验证二叉搜索树

  • 表达式树的求值

3. 后序遍历(Post-order Traversal)

访问顺序:左子树 → 右子树 → 根节点

// BinaryTree.java - 后序遍历实现 void afterOrder(TreeNode root){ if(root == null){ return; } afterOrder(root.lChild); afterOrder(root.rChild); System.out.println(root.data); }

应用场景

  • 删除二叉树

  • 计算后缀表达式

  • 计算目录大小(文件系统)

4. 层序遍历(Level-order Traversal)

按层次从上到下,从左到右访问节点

// BinaryTree.java - 层序遍历实现 void levelOrder(TreeNode root){ // 1.创建队列 LinkedList<TreeNode> queue = new LinkedList<TreeNode>(); queue.add(root); while(!queue.isEmpty()){ root = queue.pop(); System.out.println(root.data); if(root.lChild != null){ queue.add(root.lChild); } if(root.rChild != null){ queue.add(root.rChild); } } }

算法解析

  1. 使用队列作为辅助数据结构

  2. 将根节点入队

  3. 循环执行以下操作直到队列为空:

    • 出队一个节点并访问

    • 将其左子节点入队(如果存在)

    • 将其右子节点入队(如果存在)

应用场景

  • 计算二叉树深度

  • 查找最短路径

  • 按层次打印二叉树

四、节点查找辅助方法

在实现删除操作前,我们需要一些辅助方法来查找目标节点及其父节点:

// BinaryTree.java - 查找目标节点 // 找到要删除的节点 TreeNode findTarget(TreeNode root,Integer target){ if(root == null){ return null; } //去找这个值 if(root.data == target){ return root; }else if(target < root.data){ //判断是否有左子树 if(root.lChild == null){ return null; } return findTarget(root.lChild,target); }else { //判断是否有右子树 if (root.rChild == null) { return null; } return findTarget(root.rChild, target); } } //去找要删除节点的父节点 TreeNode findParent(TreeNode root,Integer target){ if(root == null){ return null; } if((root.lChild!=null) && (root.lChild.data == target) || (root.rChild!=null) && (root.rChild.data == target)){ return root; }else { if(root.lChild!=null && target < root.data){ return findParent(root.lChild,target); }else if(root.rChild!=null && target > root.data){ return findParent(root.rChild,target); }else { return null; } } }

五、有序二叉树删除节点的三种情况

删除二叉搜索树中的节点是BST操作中最复杂的部分,需要根据被删除节点的子节点数量分为三种情况处理:

情况1:删除叶子节点(没有子节点)

操作:直接删除该节点,将其父节点的对应指针设为null

实现代码

// 在delete方法中处理叶子节点的情况 if(targetNode.lChild == null && targetNode.rChild == null){ //叶子节点 //确定targetNode是parentNode的左子树还是右子树 if(parentNode.lChild != null && parentNode.lChild.data == target){ parentNode.lChild = null; }else if(parentNode.rChild != null && parentNode.rChild.data == target){ parentNode.rChild = null; } }

示例:在下图中删除节点1(叶子节点)

text

7 / \ 3 10 / \ / \ 1 5 9 11 / 0

情况2:删除只有一个子节点的节点

操作:用其子节点替换自身,保持BST性质

实现代码

// 在delete方法中处理只有一个子节点的情况 else { //targetNode 只有一个子节点的节点 //确定targetNode是parentNode的左子树还是右子树 if(parentNode.lChild!=null && parentNode.lChild.data == target){ //确定targetNode是parentNode的左子树 if(targetNode.lChild != null){ //targetNode有左子结点 parentNode.lChild = targetNode.lChild; }else { //targetNode有右子结点 parentNode.lChild = targetNode.rChild; } }else if(parentNode.rChild!=null && parentNode.rChild.data == target){ //确定targetNode是parentNode的右子树 if(targetNode.lChild != null){ //targetNode有左子结点 parentNode.rChild = targetNode.lChild; }else { //targetNode有右子结点 parentNode.rChild = targetNode.rChild; } } }

示例:在下图中删除节点1(只有一个左子节点0)

text

7 / \ 3 10 / \ / \ 1 5 9 11 / 0

情况3:删除有两个子节点的节点

操作

  1. 找到右子树中的最小节点(或左子树中的最大节点)

  2. 用这个最小(或最大)节点的值替换被删除节点的值

  3. 删除那个最小(或最大)节点(递归处理)

实现代码

// 查找右子树最小节点的辅助方法 /** * 去找右子树的最小值 * @param node * @return */ public int findRightTreeMin(TreeNode node){ while (node.lChild !=null){ node = node.lChild; } int min = node.data; delete(root, min); return min; } // 在delete方法中处理有两个子节点的情况 else if(targetNode.lChild != null && targetNode.rChild != null){ //有两个子节点的节点 int min = findRightTreeMin(targetNode.rChild); targetNode.data = min; }

示例:在下图中删除节点3(有两个子节点1和5)

text

7 / \ 3 10 / \ / \ 1 5 9 11 / 0

六、完整的删除方法实现

下面是完整的删除方法实现,整合了所有三种情况的处理:

public void delete(TreeNode root,Integer target){ if(root == null){ return; } //2.万一要删的节点只有一个节点 if(root.lChild == null && root.rChild == null){ root = null; return; } //1.去找被删除的节点 TreeNode targetNode = findTarget(root,target); if(targetNode == null){ //找不到 return; } //3.找到父节点 TreeNode parentNode = findParent(root,target); //分情况进行删除 if(targetNode.lChild == null && targetNode.rChild == null){ //叶子节点 //确定targetNode是parentNode的左子树还是右子树 if(parentNode.lChild != null && parentNode.lChild.data == target){ parentNode.lChild = null; }else if(parentNode.rChild != null && parentNode.rChild.data == target){ parentNode.rChild = null; } }else if(targetNode.lChild != null && targetNode.rChild != null){ //有两个子节点的节点 int min = findRightTreeMin(targetNode.rChild); targetNode.data = min; }else { //targetNode 只有一个子节点的节点 //确定targetNode是parentNode的左子树还是右子树 if(parentNode.lChild!=null && parentNode.lChild.data == target){ //确定targetNode是parentNode的左子树 if(targetNode.lChild != null){ //targetNode有左子结点 parentNode.lChild = targetNode.lChild; }else { //targetNode有右子结点 parentNode.lChild = targetNode.rChild; } }else if(parentNode.rChild!=null && parentNode.rChild.data == target){ //确定targetNode是parentNode的右子树 if(targetNode.lChild != null){ //targetNode有左子结点 parentNode.rChild = targetNode.lChild; }else { //targetNode有右子结点 parentNode.rChild = targetNode.rChild; } } } }

七、测试示例

// Test.java - 测试类 package com.qcby; public class Test { public static void main(String[] args) { BinaryTree bt = new BinaryTree(); bt.create(7); bt.create(3); bt.create(10); bt.create(1); bt.create(5); bt.create(9); bt.create(11); bt.create(0); System.out.println("删除节点1前的层序遍历:"); bt.levelOrder(bt.root); bt.delete(bt.root, 1); System.out.println("\n删除节点1后的层序遍历:"); bt.levelOrder(bt.root); } }

测试结果分析
原始树结构:

text

7 / \ 3 10 / \ / \ 1 5 9 11 / 0

删除节点1(只有一个左子节点0)后,节点0将取代节点1的位置:

text

7 / \ 3 10 / \ / \ 0 5 9 11

八、总结

本文通过Java代码详细讲解了二叉树的核心概念和操作:

  1. 节点定义:使用TreeNode类定义二叉树节点

  2. 二叉搜索树构建:通过create方法实现有序插入

  3. 四种遍历方法:前序、中序、后序和层序遍历的实现

  4. 节点删除的三种情况

    • 删除叶子节点:直接删除

    • 删除只有一个子节点的节点:用子节点替换

    • 删除有两个子节点的节点:用右子树最小节点替换

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/25 20:11:32

国内比较好的烘焙厨具生产商推荐榜

国内烘焙厨具生产商推荐榜&#xff1a;探寻匠心制造与全球智慧在烘焙文化日益盛行的今天&#xff0c;一套优质的烘焙厨具不仅是厨房中的得力助手&#xff0c;更是成就美味与创意的关键。中国作为全球重要的制造业基地&#xff0c;孕育了一批将传统匠心与现代工艺完美融合的优秀…

作者头像 李华
网站建设 2026/5/1 7:31:54

如何使用豆包手机的READ_FRAME_BUFFER权限截图密码画面

背景&#xff1a; 针对READ_FRAME_BUFFER的权限原理剖析前面文章已经进行了详细讲解&#xff0c;具体文章如下&#xff1a; 但是文章也给大家留下了一个疑问&#xff0c;那就是具有READ_FRAME_BUFFER权限到底是否可以截图到手机上secure画面&#xff0c;比如银行等密码手势画面…

作者头像 李华
网站建设 2026/5/1 7:41:19

31、知识探索与标注:提升信息利用效率的秘诀

知识探索与标注:提升信息利用效率的秘诀 在知识的海洋中,专家如同知识的建筑师,他们致力于理解文档所传达的知识,并用这些知识解决问题、创造新知识产品。而强大的读者在阅读时会自我监督,能察觉自身知识的缺陷并努力弥补,这体现了阅读过程中自我管理的重要性。 1. 标注…

作者头像 李华
网站建设 2026/4/30 17:52:44

33、知识管理:标签、元文档与高效学习

知识管理:标签、元文档与高效学习 在知识管理和学习过程中,标签和元文档是非常实用的工具,它们能帮助我们更好地理解、分析和应用知识资源。 标签的益处 标签具有诸多好处。它不仅能帮助我们更好地应用现有的知识架构,还能创建和完善这些架构。通过使用标签,我们可以更…

作者头像 李华
网站建设 2026/4/23 12:50:03

34、知识获取与概念分析实用指南

知识获取与概念分析实用指南 在知识的海洋中遨游,我们常常面临如何有效获取、理解和应用知识的挑战。以下将为你详细介绍知识处理、概念分析以及元文档创建与访问的实用方法。 知识处理与应用 在处理知识资源时,我们往往会产生一种错觉,认为自己能记住其中的精华并在需要…

作者头像 李华