news 2026/6/25 18:59:28

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

作者头像

张小明

前端开发工程师

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/24 3:26:27

PinWin:让窗口置顶成为你的效率倍增器

PinWin&#xff1a;让窗口置顶成为你的效率倍增器 【免费下载链接】pinwin .NET clone of DeskPins software 项目地址: https://gitcode.com/gh_mirrors/pi/pinwin 还在为窗口切换频繁打断工作节奏而苦恼吗&#xff1f;想要重要信息始终保持在视线范围内&#xff1f;Pi…

作者头像 李华
网站建设 2026/6/24 18:01:04

终极Mac菜单栏清理指南:用Dozer隐藏图标让桌面更整洁

终极Mac菜单栏清理指南&#xff1a;用Dozer隐藏图标让桌面更整洁 【免费下载链接】Dozer Hide menu bar icons on macOS 项目地址: https://gitcode.com/gh_mirrors/do/Dozer 还在为MacBook菜单栏上拥挤的图标感到烦恼吗&#xff1f;想要一个干净清爽的工作界面&#xf…

作者头像 李华
网站建设 2026/6/23 23:12:16

RocketMQ-Flink 终极实战指南:从零构建企业级实时数据管道

RocketMQ-Flink 终极实战指南&#xff1a;从零构建企业级实时数据管道 【免费下载链接】rocketmq-flink RocketMQ integration for Apache Flink. This module includes the RocketMQ source and sink that allows a flink job to either write messages into a topic or read …

作者头像 李华
网站建设 2026/6/25 15:22:34

免疫细胞去卷积分析5大核心策略:从理论到实战的完整指南

免疫细胞去卷积分析5大核心策略&#xff1a;从理论到实战的完整指南 【免费下载链接】immunedeconv 项目地址: https://gitcode.com/gh_mirrors/imm/immunedeconv 免疫细胞去卷积技术正在彻底改变我们对肿瘤微环境的理解方式。面对混合样本中细胞比例难以直接观察的困境…

作者头像 李华
网站建设 2026/6/24 23:31:33

39、高效实践:成为技能大师的秘诀

高效实践:成为技能大师的秘诀 实践若想有所回报,就需要兼具有效性和高效性。从表面上看,高效实践的过程很简单,只需接收挑战、尝试完成挑战(通常是回答问题),然后评估自己的掌握程度即可。然而,实践本身是一项复杂的技能,下面为大家提供一些通用的建议。 1. 合理安排…

作者头像 李华