news 2026/4/23 12:41:05

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

作者头像

张小明

前端开发工程师

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/20 9:10:21

MuJoCo无头渲染终极指南:云端物理仿真的高效方案

MuJoCo无头渲染终极指南&#xff1a;云端物理仿真的高效方案 【免费下载链接】mujoco Multi-Joint dynamics with Contact. A general purpose physics simulator. 项目地址: https://gitcode.com/GitHub_Trending/mu/mujoco 在当今AI与机器人技术快速发展的时代&#x…

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

Univer表格数据规范与可视化:从零到精通的5个高效技巧

Univer表格数据规范与可视化&#xff1a;从零到精通的5个高效技巧 【免费下载链接】univer Univer is a set of enterprise document and data collaboration solutions, including spreadsheets, documents, and slides. The highly extensible design allows developers to c…

作者头像 李华
网站建设 2026/4/22 15:43:24

react navite相比较传统开发有啥优势?

一、先给你「一句话结论」Web 技术栈强 → RN / Expo 追求极致性能 & 统一 UI → Flutter 快速上线 / 小团队 → Expo 复杂业务型 App → React Native 长期稳定 / 大厂级 → Flutter 或 RN&#xff08;非 Expo Go&#xff09;二、三者到底是什么关系&#xff08;很多人搞混…

作者头像 李华
网站建设 2026/4/22 17:28:14

5、提升知识应用效能:从理论到实践

提升知识应用效能:从理论到实践 1. 精益创业理念 在创业领域,很多人认为初创项目的成功依赖于天才般的创意、坚持不懈的努力和恰到好处的时机,但 Eric Ries 在其理念中指出,这种观点其实是一种误区。有些创业项目失败,是因为领导者过度信赖完美的商业计划、市场调研和预…

作者头像 李华
网站建设 2026/4/14 4:18:41

掌握n8n工作流自动化:从入门到精通的完整指南

掌握n8n工作流自动化&#xff1a;从入门到精通的完整指南 【免费下载链接】n8n n8n 是一个工作流自动化平台&#xff0c;它结合了代码的灵活性和无代码的高效性。支持 400 集成、原生 AI 功能以及公平开源许可&#xff0c;n8n 能让你在完全掌控数据和部署的前提下&#xff0c;构…

作者头像 李华