news 2026/6/22 14:19:04

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

作者头像

张小明

前端开发工程师

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/6/21 19:36:53

Spine骨骼动画与Godot集成的完整技术指南

Spine骨骼动画与Godot集成的完整技术指南 【免费下载链接】spine-runtime-for-godot This project is a module for godot that allows it to load/play Spine skeleton animation. 项目地址: https://gitcode.com/gh_mirrors/sp/spine-runtime-for-godot 在当今游戏开发…

作者头像 李华
网站建设 2026/6/17 13:37:37

DataHub数据质量监控:从入门到精通的终极指南

DataHub数据质量监控&#xff1a;从入门到精通的终极指南 【免费下载链接】datahub 项目地址: https://gitcode.com/gh_mirrors/datahub/datahub 你正在为数据质量问题而苦恼吗&#xff1f;报表频繁出错、业务决策失误、数据可信度低&#xff1f;别担心&#xff01;本文…

作者头像 李华
网站建设 2026/6/22 9:31:45

Kotaemon宏观经济数据分析:智库研究辅助工具

Kotaemon宏观经济数据分析&#xff1a;智库研究辅助工具 在当今政策节奏日益加快、经济数据瞬息万变的背景下&#xff0c;智库研究人员面临着前所未有的信息处理压力。一份关于房地产调控影响的报告&#xff0c;可能需要整合几十份部委文件、上百个城市的价格指数和多个国际机构…

作者头像 李华
网站建设 2026/6/21 21:03:38

YOLOv11n轻量化革命:如何在边缘设备上实现工业级检测精度?

YOLOv11n轻量化革命&#xff1a;如何在边缘设备上实现工业级检测精度&#xff1f; 【免费下载链接】ultralytics ultralytics - 提供 YOLOv8 模型&#xff0c;用于目标检测、图像分割、姿态估计和图像分类&#xff0c;适合机器学习和计算机视觉领域的开发者。 项目地址: http…

作者头像 李华
网站建设 2026/6/22 7:28:36

UI-TARS桌面版:用自然语言重新定义GUI自动化体验

UI-TARS桌面版&#xff1a;用自然语言重新定义GUI自动化体验 【免费下载链接】UI-TARS-desktop A GUI Agent application based on UI-TARS(Vision-Lanuage Model) that allows you to control your computer using natural language. 项目地址: https://gitcode.com/GitHub_…

作者头像 李华
网站建设 2026/6/18 20:56:31

35、信息深度挖掘与硬件配置优化指南

信息深度挖掘与硬件配置优化指南 在当今信息爆炸的时代,如何高效地从各种信息源中提取有价值的知识,并合理配置硬件环境以提升工作效率,成为了许多人关注的焦点。本文将为你详细介绍音频、电子书及其他媒体资源的深度挖掘方法,以及计算机硬件的优化配置建议。 音频资源深…

作者头像 李华