当前位置: 首页 > news >正文

《Java数据结构与算法》第四篇(三)二叉树遍历详解_CSDN文章

Java数据结构之树:二叉树的三种遍历方法详解(递归与非递归实现)

目录

  • 一、二叉树遍历的定义与重要性
  • 二、前序遍历(DLR)详解
  • 三、中序遍历(LDR)详解
  • 四、后序遍历(LRD)详解
  • 五、完整代码实现与测试
  • 六、性能对比与应用场景
  • 七、总结与学习建议

一、二叉树遍历的定义与重要性

二叉树遍历(Binary Tree Traversal)是指按照某种顺序访问二叉树中的所有节点,使得每个节点都被访问一次且仅一次。遍历是二叉树最基本、最重要的操作,是后续进行二叉树搜索、修改、删除等操作的基础。

根据访问节点的顺序不同,二叉树的遍历主要分为三种方式:

  1. 前序遍历(Preorder Traversal):根节点 → 左子树 → 右子树
  2. 中序遍历(Inorder Traversal):左子树 → 根节点 → 右子树
  3. 后序遍历(Postorder Traversal):左子树 → 右子树 → 根节点

每种遍历方式都有递归和非递归两种实现方法。递归实现简洁易懂,而非递归实现通过使用栈(Stack)数据结构来模拟递归过程,空间效率更高。

二、前序遍历(DLR)详解

2.1 前序遍历的定义

前序遍历(DLR,Data-Left-Right)是二叉树遍历中最直观的方式。其遍历规则为:

  1. 首先访问根节点
  2. 然后遍历左子树
  3. 最后遍历右子树

对于测试用例ABD##E##C##构建的二叉树:

A / \ B C / \ D E

前序遍历的结果为:A B D E C

2.2 递归实现
publicvoidDLR(BiTreeNoderoot){if(root!=null){System.out.print(root.data+" ");// 访问根节点DLR(root.lchild);// 遍历左子树DLR(root.rchild);// 遍历右子树}}

代码分析

  • 递归实现非常简洁,只有三行核心代码
  • 时间复杂度:O(n),每个节点访问一次
  • 空间复杂度:O(h),h为树的高度,递归调用栈的深度
2.3 非递归实现(数组模拟栈)
publicvoidDLR2(){BiTreeNodestack[]=newBiTreeNode[20];// 使用数组模拟栈inttop=0;BiTreeNodecurr=root;while(curr!=null||top>0){if(curr!=null){System.out.print(curr.data+" ");// 访问当前节点stack[top++]=curr;// 当前节点入栈curr=curr.lchild;// 转向左子树}if(top>0){curr=stack[--top];// 出栈curr=curr.rchild;// 转向右子树}}}
2.4 非递归实现(Java Stack类)
publicStringDLR3(){StringBuilderresult=newStringBuilder();if(root==null){return"";}Stack<BiTreeNode>stack=newStack<>();stack.push(root);// 根节点入栈while(!stack.isEmpty()){BiTreeNodecurr=stack.pop();result.append(curr.data+" ");// 访问当前节点// 右子树先入栈(后处理)if(curr.rchild!=null){stack.push(curr.rchild);}// 左子树后入栈(先处理)if(curr.lchild!=null){stack.push(curr.lchild);}}returnresult.toString();}

注意:这里右子树先入栈,左子树后入栈,因为栈是后进先出(LIFO)的数据结构,这样才能保证先处理左子树。

三、中序遍历(LDR)详解

3.1 中序遍历的定义

中序遍历(LDR,Left-Data-Right)的特点是:

  1. 首先遍历左子树
  2. 然后访问根节点
  3. 最后遍历右子树

对于同一棵二叉树,中序遍历的结果为:D B E A C

重要特性:对于二叉搜索树(BST),中序遍历会得到有序的节点序列。

3.2 递归实现
publicvoidLDR(BiTreeNoderoot){if(root!=null){LDR(root.lchild);// 先遍历左子树System.out.print(root.data);// 再访问根节点LDR(root.rchild);// 最后遍历右子树}}
3.3 非递归实现
publicStringLDR2(){StringBuilderresult=newStringBuilder();if(root==null){return"";}Stack<BiTreeNode>stack=newStack<>();BiTreeNodecurr=root;while(!stack.isEmpty()||curr!=null){// 一直向左走到底while(curr!=null){stack.push(curr);curr=curr.lchild;}// 弹出栈顶节点并访问curr=stack.pop();result.append(curr.data+" ");// 转向右子树curr=curr.rchild;}returnresult.toString();}

算法思路

  1. 从根节点开始,将路径上的所有节点入栈,直到最左边的叶子节点
  2. 弹出栈顶节点并访问
  3. 转向该节点的右子树,重复上述过程

四、后序遍历(LRD)详解

4.1 后序遍历的定义

后序遍历(LRD,Left-Right-Data)的顺序为:

  1. 首先遍历左子树
  2. 然后遍历右子树
  3. 最后访问根节点

对于同一棵二叉树,后序遍历的结果为:D E B C A

应用场景:后序遍历常用于需要先处理子节点再处理父节点的场景,如计算目录大小、释放树形结构内存等。

4.2 递归实现
publicvoidLRD(BiTreeNoderoot){if(root!=null){LRD(root.lchild);// 先遍历左子树LRD(root.rchild);// 再遍历右子树System.out.print(root.data);// 最后访问根节点}}
4.3 非递归实现(双栈法)
publicStringLRD2(){StringBuilderresult=newStringBuilder();if(root==null){return"";}Stack<BiTreeNode>stack1=newStack<>();// 辅助栈Stack<BiTreeNode>stack2=newStack<>();// 结果栈stack1.push(root);while(!stack1.isEmpty()){BiTreeNodecurr=stack1.pop();stack2.push(curr);// 将节点放入结果栈// 左子树先入栈if(curr.lchild!=null){stack1.push(curr.lchild);}// 右子树后入栈if(curr.rchild!=null){stack1.push(curr.rchild);}}// 从结果栈中弹出得到后序序列while(!stack2.isEmpty()){BiTreeNodecurr=stack2.pop();result.append(curr.data+" ");}returnresult.toString();}
4.4 非递归实现(单栈法)
publicStringLRD3(){StringBuilderresult=newStringBuilder();if(root==null){return"";}Stack<BiTreeNode>stack=newStack<>();BiTreeNodecurr=root;BiTreeNodeprev=null;// 记录上一个访问的节点while(!stack.isEmpty()||curr!=null){if(curr!=null){stack.push(curr);curr=curr.lchild;}else{BiTreeNodetemp=stack.peek();// 如果右子树存在且未被访问if(temp.rchild!=null&&prev!=temp.rchild){curr=temp.rchild;}else{// 访问该节点result.append(temp.data+" ");prev=stack.pop();}}}returnresult.toString();}

五、完整代码实现与测试

5.1 二叉树节点类
classBiTreeNode{chardata;// 节点数据BiTreeNodelchild,rchild;// 左右孩子指针// 默认构造函数publicBiTreeNode(){}// 带参数的构造函数publicBiTreeNode(chardata){this.data=data;lchild=null;rchild=null;}// 完整构造函数publicBiTreeNode(chardata,BiTreeNodelchild,BiTreeNoderchild){this.data=data;this.lchild=lchild;this.rchild=rchild;}}
5.2 二叉树的构建
publicvoidcreateBiTree(Stringinput){pi=0;root=createBiTreeHelper(input);num=countNodes(root);}privateBiTreeNodecreateBiTreeHelper(Stringinput){if(pi>=input.length()||input.charAt(pi)=='#'){pi++;returnnull;// #表示空节点}BiTreeNoderoot=newBiTreeNode(input.charAt(pi));++pi;root.lchild=createBiTreeHelper(input);// 递归构建左子树root.rchild=createBiTreeHelper(input);// 递归构建右子树returnroot;}

构建规则:使用先序序列和特殊字符#来表示空节点,如ABD##E##C##

5.3 运行结果测试

运行截图展示了三种遍历方式的测试结果:

测试用例构建的二叉树结构:

A / \ B C / \ D E

六、性能对比与应用场景

6.1 时间空间复杂度对比
遍历方法时间复杂度空间复杂度(递归)空间复杂度(非递归)
前序遍历O(n)O(h)O(h)
中序遍历O(n)O(h)O(h)
后序遍历O(n)O(h)O(h)

其中n为节点数,h为树的高度。最坏情况下(树退化为链表),h = n。

6.2 非递归实现的优势
  1. 空间效率更高:避免了递归调用的开销
  2. 不会栈溢出:递归深度过深时可能导致栈溢出
  3. 更好的控制:可以在遍历过程中进行更灵活的操作
6.3 应用场景
  • 前序遍历:复制树结构、表达式树求值
  • 中序遍历:二叉搜索树的中序输出(有序序列)
  • 后序遍历:计算表达式值、释放树内存、文件系统遍历

七、总结与学习建议

7.1 核心要点总结
  1. 理解遍历本质:二叉树遍历是将树形结构线性化的过程
  2. 掌握递归思想:递归实现简洁直观,是理解遍历的基础
  3. 理解栈的作用:非递归实现通过栈模拟递归调用过程
  4. 注意特殊情况:空树、单节点树等边界条件
  5. 选择合适方法:根据实际需求选择递归或非递归实现
7.2 学习建议
  1. 画图辅助理解:手动画出遍历路径,加深理解
  2. 调试跟踪过程:使用IDE调试功能跟踪遍历过程
  3. 多种实现方式:掌握同一遍历的不同实现方法
  4. 实际应用练习:结合实际问题练习遍历应用
7.3 扩展学习

二叉树遍历是树形结构的基础,建议继续学习:

  • 层次遍历(广度优先搜索)
  • 线索二叉树
  • 平衡二叉树(AVL树)
  • 红黑树
  • B树和B+树

参考资源:

  • Java官方文档 - Stack类
  • 数据结构与算法分析
  • 算法可视化网站

标签:#Java数据结构 #二叉树 #树遍历 #算法实现 #数据结构基础

如果这篇文章对你有帮助,欢迎点赞、收藏和评论!有疑问的小伙伴可以在评论区留言交流。

http://www.cnnetsun.cn/news/139767.html

相关文章:

  • MindSpore硬核实战:彻底搞懂自动混合精度(AMP)与函数式训练
  • Java异常处理详解。零基础小白到精通,收藏这篇就够了
  • 基于深度学习YOLOv12的犬种识别检测系统(YOLOv12+YOLO数据集+UI界面+登录注册界面+Python项目源码+模型)
  • 基于深度学习YOLOv11的犬种识别检测系统(YOLOv11+YOLO数据集+UI界面+登录注册界面+Python项目源码+模型)
  • [插电式混合动力车辆][交替方向乘子法(ADMM)结合CVX]插电式混合动力车辆的能源管理:基于凸优化算法用于模型预测控制MPC研究附Matlab代码
  • 【别花冤枉钱】学生党专享!2025年把AI率90%降到10%的“低成本”组合拳(含免费/付费工具避坑指南)
  • 前端Vue制作日历插件FullCalendar,零基础入门到精通,收藏这篇就够了
  • 基于MPC算法的P2构型混合动力汽车能量管理优化策略
  • 德克萨斯大学奥斯汀分校突破:球形利奇量化提升AI图像生成质量
  • 13、Unix 系统管理脚本实用指南(上)
  • 2026网络安全薪酬全景:哪些岗位是价值洼地,哪里又是薪资天花板?
  • Oracle领衔科技巨头5000亿美元AI数据中心租赁狂潮
  • Java算法——排序篇之快速排序,零基础小白到精通,收藏这篇就够了
  • 平安好医生:“人+机+生态”闭环 打造中国AI医疗标杆
  • Compose 适配 - 全屏显示 EdgeToEdge
  • python-flask-django重症监护室中急诊护理管理系统设计与实现_zjv2nt1d
  • 拿一句,逗得你家男人哭笑不得
  • 虎贲等考 AI:AI 赋能学术全流程,让论文写作从 “煎熬” 到 “高效”✨
  • 介观交通流仿真软件:VISSIM (介观模式)_(5).车辆行为模型
  • 英特尔酷睿Ultra第三代,如何推动AI PC规模化落地?
  • 15、密码学编程问题与解决方案
  • 【花雕学编程】Arduino BLDC 之基础差速转向小车(串口控制)
  • 【毕业设计】基于springboot+Android的研学旅行服务平台APP小程序设计(源码+文档+远程调试,全bao定制等)
  • 应用——管道与文件描述符
  • 【总结】【数据结构】【OS】【计组】【计网】
  • 小程序毕设项目:基于springboot的智能学习小程序(源码+文档,讲解、调试运行,定制等)
  • 小程序毕设项目:基于springboot+微信小程序的大学生餐厅点餐系统小程序(源码+文档,讲解、调试运行,定制等)
  • Flutter 与 AI 深度集成:用 Gemini 打造智能应用的实战指南(2025 版)
  • 零基础IM开发入门:什么是IM聊天系统的端到端加密?
  • MyBatis批量插入从5分钟优化到3秒,我做了这3件事