数据结构(Java):力扣 二叉树面试OJ题(二)【进阶】

目录

💎 1、题一:二叉树的层序遍历

🌟 1.1 思路1(递归求解) 

🌟 1.1.1 思路1代码

🔆 1.2 思路2(队列求解)

🔆 1.2.1 思路2代码 

 💎 2、题二:判断是否为完全二叉树

🌟 2.1 思路分析

  🌟 2.2 代码

💎  3、题三:求二叉树中节点的最近公共祖先 

🌟 3.1 思路分析

 🌟 3.2 代码

💎  4、题四:根据二叉树创建字符串

🌟 4.1 思路分析

 🌟 4.2 代码

💎  5、题五:二叉树前序遍历递归实现与非递归实现

🌟 5.1 递归实现->思路分析

 🌟 5.1.1 递归实现->代码

🔆 5.2 非递归实现->思路分析

​编辑 🔆 5.2.1 非递归实现->代码 

💎  6、题六:二叉树中序遍历递归实现与非递归实现

🌟 6.1 递归实现->代码

🔆 6.2 非递归实现->代码

 💎  7、题七:二叉树后序遍历递归实现与非递归实现

🌟 7.1 递归实现->代码

🔆 7.2 非递归实现->思路分析

 🔆 7.2.1 非递归实现->代码 


💎 1、题一:二叉树的层序遍历

. - 力扣(LeetCode)

层序遍历, 即从上到下,从左到右,一层一层的将数据遍历一遍。 

🌟 1.1 思路1(递归求解) 

  1. 首先,根据上篇求树高度的方法,求出二叉树的高度。接着,实例化出高度个一维顺序表添加到二维顺序表中。
  2. 记录好每层数据的层数,根据层数并按顺序来将来将数据添加到二维顺序表的第几行中(记录层数来放入数据,是解题关键点!!!)。
  3. 默认根节点是第0层,与顺序表下标相对应。
  4. 这里以前序遍历的思想递归,添加各层数据。

🌟 1.1.1 思路1代码

class Solution {public List<List<Integer>> levelOrder(TreeNode root) {List<List<Integer>> listVal = new ArrayList<>();if(root == null) {//注意:若根节点为空,要返回空的顺序表return listVal;};int height = treeHeight(root);for(int i = 0; i < height; i++) {List<Integer> list = new ArrayList<>();listVal.add(list);}//根节点的层数默认为0addNode(root,0,listVal);return listVal;}//求树高度public int treeHeight(TreeNode root) {if(root == null) return 0;return Math.max(treeHeight(root.left),treeHeight(root.right)) + 1;}public void addNode(TreeNode root,int level,List<List<Integer>> listVal) {if(root == null) {return;}//以前序遍历的思想,根据层数来将数据添加到该行的顺序表中listVal.get(level).add(root.val);addNode(root.left,level+1,listVal);addNode(root.right,level+1,listVal);} 
}


🔆 1.2 思路2(队列求解)

  1. 首先创建一个队列,将根节点放入队列当中
  2. 每次循环前记录队列中元素个数size,依次出队size个元素,每次出队的元素(节点)都要判断其左右子树是否为空,若不为空则添加到队列中
  3. 将每次出队的元素的val值添加到一维顺序表中,size个元素全部出队完后,将这个一维顺序表添加到二维顺序表中,也就是说,每size个出队的元素都是同一层的节点
  4. 循环以上过程,直至元素全部添加至二维顺序表中(队列为空

🔆 1.2.1 思路2代码 

class Solution {public List<List<Integer>> levelOrder(TreeNode root) {List<List<Integer>> listVal = new ArrayList<>();if (root == null) {//注意:若根节点为空,要返回空的顺序表return listVal; }Queue<TreeNode> queue = new LinkedList<>();queue.offer(root);//队列空时,说明节点已全部添加到二维顺序表中while (!queue.isEmpty()) {int size = queue.size();//存储每一层节点的顺序表List<Integer> list = new ArrayList<>();//将队列中的size个元素依次出队,//并判断其左右子树是否为空,若不为空则添加到队列中while (size != 0) {TreeNode out = queue.poll();list.add(out.val);if (out.left != null) {queue.offer(out.left);}if (out.right != null) {queue.offer(out.right);}size--;}//将每一层的一维顺序表添加到二维顺序表中listVal.add(list);}return listVal;}
}

 💎 2、题二:判断是否为完全二叉树

 本题无链接,

题目即:给出一棵树的根节点,判断是否为完全二叉树。

🌟 2.1 思路分析

这题的解题思路和上一题队列求解差不多,都是借助队列这一数据结构:

  1. 首先,创建队列,放入根节点。
  2. 将一个元素出队,cur接收出队元素,判断cur是否为空,若cur不为空,则将其左右孩子放入队列中(若左右子树为null,也要将null入队),循环这一过程。
  3. 当cur为null时结束循环(出队的是null,cur接收,此时cur为null),此时,只要判断队列中的剩余元素是否存在非null元素:若剩余元素全为null则说明该树为完全二叉树;若剩余元素存在非null元素则说明该树不为完全二叉树。

  🌟 2.2 代码

boolean isCompleteTree(TreeNode root) {if(root == null) {return true;}//创建队列 添加根节点Queue<TreeNode> queue = new LinkedList<>();queue.offer(root);//当cur不为空时,添加其左右孩子TreeNode cur = root;while (cur != null) {cur = queue.poll();if (cur != null) {queue.offer(cur.left);queue.offer(cur.right);}}//cur为空时,判断队列中是否有非空元素//若剩余元素全为null则说明该树为完全二叉树;// 若剩余元素存在非null元素则说明该树不为完全二叉树。boolean flg = false;while (!queue.isEmpty()) {TreeNode cur2 = queue.poll();if (cur2 != null) {flg = true;}}if (flg == true) {return false;}else {return true;}}

💎  3、题三:求二叉树中节点的最近公共祖先 

. - 力扣(LeetCode)

🌟 3.1 思路分析

  • ①:当p、q在根节点的左右子树上时,祖先就是这个根节点
  • ②:当p、q都在根节点的右子树上时,由于节点的祖先可以是它自己,所以祖先就是p或q本身
  • ③:当p、q都在根节点的左子树上时,由于节点的祖先可以是它自己,所以祖先就是p或q本身
  • 以前序遍历思想递归,若遇见p、q节点便返回
  • 若左右子树都有返回值,说明祖先就是这个根节点;(对于第一层的根节点)若只有左子树有返回值,说明p、q均在左子树上,返回的节点就是最近祖先;(对于第一层的根节点)若只有右子树有返回值,说明p、q均在右子树上,返回的节点就是最近祖先。

 🌟 3.2 代码

class Solution {public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {return findNode( root,  p,  q);}public TreeNode findNode(TreeNode root, TreeNode p, TreeNode q) {if(root == null) {return null;}//先判断根节点是否为p、q节点if(root.val == p.val || root.val == q.val) {return root;}//查找左右子树TreeNode leftNode = findNode(root.left,p,q);TreeNode rightNode = findNode(root.right,p,q);   //若左右子树都有返回值,说明祖先就是这个根节点    if(leftNode != null && rightNode != null) {return root;}//在左树上找到了if(leftNode != null && rightNode == null) {return leftNode;}//在右树上找到了if(rightNode != null && leftNode == null) {return rightNode;}return null;}
}

💎  4、题四:根据二叉树创建字符串

. - 力扣(LeetCode)

🌟 4.1 思路分析

  1. 借助StringBuffer拼接字符串,最后返回字符串
  2. 以前序遍历思想递归
  3. 拼接当前根节点,判断根节点的左孩子是否为空,若左孩子不为空则拼接"(",并递归左子树;若左孩子为空且右孩子为空,则直接返回即可;若左孩子为空但右孩子不为空,注意!!!此时应该拼接"()",继续递归右子树。
  4. 若右孩子为空,则直接返回;若右孩子不为空,则拼接"("并递归右子树。
  5. 在根节点的左右子树完成即递归回退后拼接相应的")"
  6. 调用toString方法返回字符串

 🌟 4.2 代码

 class Solution {StringBuilder stringBuilder = new StringBuilder();public String tree2str(TreeNode root) {tree2strChild(root);return stringBuilder.toString();}public void tree2strChild(TreeNode root) {if(root == null) {return ;}stringBuilder.append(root.val);if(root.left != null) {//左子树不为空,则必然添加"("stringBuilder.append("(");//递归左子树tree2strChild(root.left);//该节点左子树完,则拼接")"stringBuilder.append(")");}else {//左子树为空,且右子树为空,直接返回if(root.right == null) {return;}else {//左子树为空,但右子树不为空,需拼接"()"stringBuilder.append("()");}}//若右孩子为空,则直接返回;若右孩子不为空,则拼接"("并递归右子树if (root.right != null) {stringBuilder.append("(");tree2str(root.right);stringBuilder.append(")");}else {return;}}
}

💎  5、题五:二叉树前序遍历递归实现与非递归实现

 . - 力扣(LeetCode)

🌟 5.1 递归实现->思路分析

递归实现前序遍历的思想很简单:就是先将根节点的数据放入顺序表中,再将左子树的数据放入顺序表中, 再将右子树的数据放入顺序表中。

 🌟 5.1.1 递归实现->代码

class Solution {public List<Integer> preorderTraversal(TreeNode root) {List<Integer> list = new ArrayList<>();if(root == null) {return list;}list.add(root.val);List<Integer> leftList = preorderTraversal(root.left);list.addAll(leftList);List<Integer> rightList = preorderTraversal(root.right);list.addAll(rightList);return list;}
}

🔆 5.2 非递归实现->思路分析

 非递归实现遍历,我们需要借助一种数据结构--->

注意:本题要求将数值放入顺序表中,返回顺序表。故这里访问根节点时,对根节点的操作是将根节点的val值添加到顺序表中。 

  1. 创建栈和顺序表,将根节点赋值给cur
  2. 将cur入栈并将cur的val值添加到顺序表中,cur更新为cur的left,接着将cur入栈并将cur的val值添加到顺序表中。循环操作,直到cur为空
  3. 此时cur为空,pop弹出栈顶元素赋给top,将cur更新为top.right;若top.right为空,即更新的cur为空,则继续弹出栈顶元素(top接收),直至cur不为空。
  4. 此时cur不为空,循环2、3步骤,直至栈中元素为空,说明树中元素已按照前序遍历的方式全部添加到了顺序表中。

 🔆 5.2.1 非递归实现->代码 

class Solution {public List<Integer> preorderTraversal(TreeNode root) {List<Integer> list = new ArrayList<>();Stack<TreeNode> stack = new Stack<>();TreeNode cur = root;//注意这里的循环条件//因为后续cur为空时要弹出栈顶元素,故栈不能为空//且一开始时栈为空,故cur不为空就可进入循环//当栈空且cur为空,说明前序遍历已完成while (cur != null || !stack.isEmpty()) {//cur不为空就要进栈,并添加进顺序表//cur = cur.leftwhile (cur != null) {stack.push(cur);list.add(cur.val);cur = cur.left;}//当cur为空时,弹出栈顶元素(top),赋给cur其右子树//若cur还是为空(top.right为空),循环过来继续弹出元素,并赋给cur新的top.rightTreeNode top = stack.pop();cur = top.right;}return list;}
}

💎  6、题六:二叉树中序遍历递归实现与非递归实现

. - 力扣(LeetCode)

中序遍历的实现,不管是在递归实现上还是在非递归实现上,和前序遍历的思想都是相同的,只是访问根节点的时机有所不同。

🌟 6.1 递归实现->代码

class Solution {public List<Integer> inorderTraversal(TreeNode root) {List<Integer> list = new ArrayList<>(); if(root == null) {return list;}List<Integer> listLeft = inorderTraversal(root.left);list.addAll(listLeft);list.add(root.val);List<Integer> listRight = inorderTraversal(root.right);list.addAll(listRight);return list;}
}

🔆 6.2 非递归实现->代码

class Solution {public List<Integer> inorderTraversal(TreeNode root) {List<Integer> list = new ArrayList<>();Stack<TreeNode> stack = new Stack<>();TreeNode cur = root;//注意这里的循环条件//因为后续cur为空时要弹出栈顶元素,故栈不能为空//且一开始时栈为空,故cur不为空就可进入循环//当栈空且cur为空,说明前序遍历已完成while (cur != null || !stack.isEmpty()) {//cur走左子树while (cur != null) {stack.push(cur);cur = cur.left;}//当cur为空时,弹出栈顶元素(top),添加进顺序表,赋给cur其右子树//若cur还是为空(top.right为空),循环过来继续弹出元素,并赋给cur新的top.rightTreeNode top = stack.pop();list.add(top.val);cur = top.right;}return list;}
}

 💎  7、题七:二叉树后序遍历递归实现与非递归实现

. - 力扣(LeetCode)

 后序遍历的实现,在递归上,与前序、中序遍历的思想是相同的,在此不再赘述。

而在非递归的实现上,后序遍历需要额外注意几点。

🌟 7.1 递归实现->代码

class Solution {public List<Integer> postorderTraversal(TreeNode root) {List<Integer> list = new ArrayList<>();if(root == null) {return list;}List<Integer> list1 = postorderTraversal(root.left);list.addAll(list1);List<Integer> list2 = postorderTraversal(root.right);list.addAll(list2);list.add(root.val);return list;}
}

🔆 7.2 非递归实现->思路分析

 后序遍历在非递归的实现上,需要额外注意几点:

  1. 当cur为空时,不可直接将栈顶元素弹出并添加进顺序表(或打印,只是访问根节点时的操作不同)
  2. 因为栈顶元素即根节点,后序遍历,需要判断栈顶元素的右孩子是否为空,若为空,才可弹出栈顶元素并打印;若右孩子不为空,需要先对右子树中的元素进行入栈和打印操作
  3. 而当栈顶元素的右孩子不为空时,我们又需要额外注意一点!!!当栈顶元素的右孩子不为空时,那就说明这个栈顶元素将永远不能出栈,会造成死循环!!!我们需要记录进过栈的元素,避免死循环的出现!!! 

发生死循环过程如下图:

 🔆 7.2.1 非递归实现->代码 

class Solution {public List<Integer> postorderTraversal(TreeNode root) {List<Integer> list = new ArrayList<>();Stack<TreeNode> stack = new Stack<>();TreeNode cur = root;//prev指向被打印过的节点TreeNode prev = null;//注意这里的循环条件//因为后续cur为空时要弹出栈顶元素,故栈不能为空//且一开始时栈为空,故cur不为空就可进入循环//当栈空且cur为空,说明前序遍历已完成while (cur != null || !stack.isEmpty()) {//cur走左子树while (cur != null) {stack.push(cur);cur = cur.left;}//当cur为空时,不可直接将栈顶元素弹出和添加进顺序表//要判断栈顶元素(根节点)是否有右孩子//若右孩子为空可以弹出和添加//若右孩子不为空,需要先将右子树的元素入栈并添加TreeNode top = stack.peek();//当栈顶元素的右孩子入过栈时,此时栈顶元素可以弹出并添加进顺序表if(top.right == null || top.right == prev) {stack.pop();list.add(top.val);prev = top;}else {cur = top.right;}}return list;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/3247524.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

基于Java中的SSM框架实现求职招聘网站系统项目【项目源码】

基于Java中的SSM框架实现线求职招聘网站系统演示 研究方法 本文的研究方法主要有&#xff1a; &#xff08;1&#xff09;调查法 调查法就是在系统的构思阶段&#xff0c;设计者对系统的功能和系统的现状有些不了解&#xff0c;需要去实地的去和本系统相关的区域进行调查&am…

制造运营管理系统(MOM系统),企业实现先进制造的关键一步

随着全球制造业的快速发展&#xff0c;企业对于生产效率和成本控制的要求日益增高。在这个背景下&#xff0c;制造运营管理系统&#xff08;MOM系统&#xff09;成为了企业提升竞争力的关键工具。盘古信息作为业内领先的智能制造解决方案提供商&#xff0c;其MOM系统更是以其卓…

django学习入门系列之第四点《BootStrap依赖》

文章目录 往期回顾 BootStrap依赖于JavaScript的类库&#xff0c;JQuery下载 下载JQuery&#xff0c;在界面上应用JQuery 在页面上应用BootStrap和avaScript的类库【JQuery是avaScript的类库】 JQuery的官网&#xff1a; jQuery 如果要应用JQuery 则要在body里面导入文件…

华为HCIP Datacom H12-821 卷42

42.填空题 如图所示&#xff0c;MSTP网络中SW1为总根&#xff0c;请将以下交换机与IST域根和主桥配对。 参考答案&#xff1a;主桥1468 既是IST域根又是主桥468 既不是又不是就是25 解析&#xff1a; 主桥1468 既是IST域根又是主桥468 既不是又不是就是25 43.填空题 网络有…

【漏洞复现】泛微OA E-Cology getdata.jsp SQL注入漏洞

免责声明&#xff1a; 本文内容旨在提供有关特定漏洞或安全漏洞的信息&#xff0c;以帮助用户更好地了解可能存在的风险。公布此类信息的目的在于促进网络安全意识和技术进步&#xff0c;并非出于任何恶意目的。阅读者应该明白&#xff0c;在利用本文提到的漏洞信息或进行相关测…

Spring Boot集成kudu快速入门Demo

1.什么是kudu 在Kudu出现前&#xff0c;由于传统存储系统的局限性&#xff0c;对于数据的快速输入和分析还没有一个完美的解决方案&#xff0c;要么以缓慢的数据输入为代价实现快速分析&#xff0c;要么以缓慢的分析为代价实现数据快速输入。随着快速输入和分析场景越来越多&a…

【VScode】安装【ESP-IDF】插件及相关工具链

一、ESP-IDF简介 二、VScode安装ESP-IDF插件 三、安装ESP-IDF、ESP-IDF-Tools以及相关工具链 四、测试例程&编译烧录 一、ESP-IDF简介 二、VScode安装ESP-IDF插件 【VScode】安装配置、插件及远程SSH连接 【VSCode】自定义配置 打开VScode&#xff0c;在插件管理搜索esp…

视频共享融合赋能平台LntonCVS视频监控业务平台技术方案详细介绍

LntonCVS国标视频综合管理平台是一款智慧物联应用平台&#xff0c;核心技术基于视频流媒体&#xff0c;采用分布式和负载均衡技术开发&#xff0c;提供广泛兼容、安全可靠、开放共享的视频综合服务。该平台功能丰富&#xff0c;包括视频直播、录像、回放、检索、云存储、告警上…

水利行业的智慧革命:深度剖析智慧水利解决方案,看其如何以科技力量提升水资源管理效率,保障水生态安全

目录 一、智慧水利的概念与内涵 二、智慧水利解决方案的核心要素 1. 感知层&#xff1a;全面监测&#xff0c;精准感知 2. 网络层&#xff1a;互联互通&#xff0c;信息共享 3. 平台层&#xff1a;数据分析&#xff0c;智能决策 4. 应用层&#xff1a;精准施策&#xff0…

构建gitlab远端服务器(check->build->test->deploy)

系列文章目录 提示:这里可以添加系列文章的所有文章的目录,目录需要自己手动添加 TODO:写完再整理 文章目录 系列文章目录前言构建gitlab远端服务器一、步骤一:搭建gitlab的运行服务器【运维】1. 第一步:硬件服务器准备工作(1)选择合适的硬件和操作系统linux(2)安装必…

C语言 | Leetcode C语言题解之第240题搜索二维矩阵II

题目&#xff1a; 题解&#xff1a; bool searchMatrix(int** matrix, int matrixSize, int* matrixColSize, int target){int i 0;int j matrixColSize[0] - 1;while(j > 0 && i < matrixSize){if(target < matrix[i][j])j--;else if(target > matrix[…

用户注册业务逻辑、接口设计和实现、前端逻辑

一、用户注册业务逻辑分析 二、用户注册接口设计和定义 2.1. 设计接口基本思路 对于接口的设计&#xff0c;我们要根据具体的业务逻辑&#xff0c;设计出适合业务逻辑的接口。设计接口的思路&#xff1a; 分析要实现的业务逻辑&#xff1a; 明确在这个业务中涉及到几个相关子…

【GraphRAG】微软 graphrag 效果实测

GraphRAG 本文将基于以下来源&#xff0c;对Microsoft GraphRAG分析优缺点、以及示例实测分析。 1. Source 代码仓库&#xff1a; Welcome to GraphRAGhttps://microsoft.github.io/graphrag/ 微软文章1&#xff08;2024.2.13&#xff09;&#xff1a;GraphRAG: Unlocking…

Web开发:四角线框效果(HTML、CSS、JavaScript)

目录 一、实现效果 二、完整代码 三、页面准备 1、页面结构 2、初始样式 3、现有效果 三、线框实现 1、需求分析 2、线框结构 3、线框大小 4、线框位置 5、线框样式 6、移动线框 7、添加过渡效果 8、使用CSS变量 一、实现效果 如下图所示&#xff0c;当鼠标移动…

微服务设计原则——高性能:锁

文章目录 1.锁的问题2.无锁2.1 串行无锁2.2 无锁数据结构 3.减少锁竞争参考文献 1.锁的问题 高性能系统中使用锁&#xff0c;往往带来的坏处要大于好处。 并发编程中&#xff0c;锁带解决了安全问题&#xff0c;同时也带来了性能问题&#xff0c;因为锁让并发处理变成了串行操…

WebStorm 2024.1 最新变化 附问卷调查 AI

WebStorm 2024.1 最新变化 问卷调查项目在线AI WebStorm 2024.1 最新变化关键亮点粘性行快速文档改进全行代码补全默认启用的 Vue Language Server适用于 Vue、Svelte 和 Astro 的组件用法*Language Services*&#xff08;语言服务&#xff09;微件 框架和技术实验性 TypeScrip…

【PPT笔记】1-3节 | 默认设置/快捷键/合并形状

文章目录 说明笔记1 默认设置1.1 OFFICE版本选择1.1.1 Office某某数字专属系列1.1.2 Office3651.1.3 产品信息怎么看 1.2 默认设置1.2.1 暗夜模式1.2.2 无限撤回1.2.3 自动保存&#xff08;Office2013版本及以上&#xff09;1.2.4 图片压缩1.2.5 字体嵌入1.2.6 多格式导出1.2.7…

arcgis怎么选取某个指定区域地方的数据,比如从全国乡镇数据选取长沙市乡镇数据

一共5个步骤&#xff0c;没一句废话&#xff0c;耐心看完。 1、如图&#xff0c;先将数据加载到arcgis里面&#xff0c;我们要选取里面长沙市的范围数据。 2、选取长沙市的语句 “市” like ‘长沙%’ 切记&#xff0c;切记&#xff0c;切记。所有符号要在 输入法英文状态…

5.5 软件工程-系统测试

系统测试 - 意义和目的 系统测试 - 原则 系统测试 - 测试过程 系统测试 - 测试策略 系统测试 - 测试方法 真题 系统测试 - 测试用例设计 黑盒测试 白盒测试 真题 系统测试 - 调试 系统测试 - 软件度量 真题

服务器IP和电脑IP有什么不同

服务器IP和电脑IP有什么不同&#xff1f;在当今的信息化时代&#xff0c;IP地址作为网络世界中不可或缺的元素&#xff0c;扮演着举足轻重的角色。然而&#xff0c;对于非专业人士来说&#xff0c;服务器IP和电脑IP之间的区别往往模糊不清。本文旨在深入探讨这两者之间的不同&a…