数据结构(Java):力扣牛客 二叉树面试OJ题(一)

👉 ​​​​​​目录 👈

1、题一:检查两棵树是否相同

 1.1 思路分析

1.2 代码

2、题二:另一棵树的子树

2.1 思路分析

 2.2 代码

3、题三:翻转二叉树

3.1 思路分析

3.2 代码

4、题四:判断树是否对称

4.1 思路分析

 4.2 代码

 5、题五:判断是否为平衡二叉树

5.1 思路分析

5.1.1 平衡二叉树概念

5.1.2 思路一  O(n^2)

5.1.3 思路一代码 

 5.1.4 思路二 :改善为O(n)【字节跳动面试题】

 5.1.5 思路二代码

6、题六:将二叉搜索树转化为有序的双向链表

6.1 二叉搜索树的性质

6.2  思路分析

6.3 代码

7、题七:二叉树的遍历和构建

7.1  思路分析

​7.2 代码


1、题一:检查两棵树是否相同

. - 力扣(LeetCode)

 1.1 思路分析

两棵树相同,需要注意以下几点:

  1. 两棵树结构相同,若结构不相同,那两棵树必然是不相同的
  2. 在结构相同的前提下,还需满足节点值相同
  3. 若结构和节点值都相同,那么两棵树相同

额外注意的是:若两棵树为空,那么也是相同的。

 我们可以按照子问题遍历的思想,当左子树和右子树中全部节点同时满足以上条件,说明两棵树相等。

1.2 代码

时间复杂度:O(min(m,n))

class Solution {public boolean isSameTree(TreeNode p, TreeNode q) {//如果两棵树都为空,那么两棵树相同if(p == null && q == null) {return true;}//如果两棵树的结构不同,那这两棵树必然不同if(p != null && q == null || p == null && q != null) {return false;}//走到这里,说明两棵树的结构相同且不为空,那判断他们的节点值即可//若节点值不相同,则说明两棵树不相同,返回falseif(p.val != q.val) {return false;}//递归遍历//左右两棵子树必须同时满足条件 //才能说明两棵树才相同return isSameTree(p.left,q.left) &&isSameTree(p.right,q.right);}
}

2、题二:另一棵树的子树

. - 力扣(LeetCode)

2.1 思路分析

一棵树的子树 需要满足:

  1. 这棵子树包括主树的某个节点,和这个节点的所有后代节点
  2. 这棵树自身,也可看做自己的子树

了解子树概念后,我们可以按照以下思路解决问题:

  1. 判断子树根节点subRoot和主树根节点root是否相同
  2. 若相同,则调用检查两棵树是否相同的方法(题一),若相同,则为子树
  3. 若不相同,继续遍历,判断子树是否和root的左子树相同 
  4. 若不相同,继续遍历,判断子树是否和root的右子树相同
  5. 递归解决问题 

 2.2 代码

 时间复杂度:O(m*n)

 因为最坏的情况下:

 每个节点和子树值相同,但判断树是否相同时,总是最后一个节点不相同

 /**时间复杂度为:O(m*n)因为最坏的情况下:每个节点和子树值相同,但判断树是否相同时,总是最后一个节点不相同
*/
class Solution {public boolean isSubtree(TreeNode root, TreeNode subRoot) {//如果递归到空,说明没有找到子树,返回falseif(root == null) {return false;}//判断当前节点的整颗树和子树是否相同即可if(isSameTree(root,subRoot)) {return true;}//如果递归遇见了子树 则一路返回trueif(isSubtree(root.left,subRoot)) return true;if(isSubtree(root.right,subRoot)) return true;//本次递归没有找见子树 返回falsereturn false;}public boolean isSameTree(TreeNode p, TreeNode q) {// 1.先判断结构是否是一样的if (p != null && q == null || p == null && q != null) {return false;}// 上述if语句 如果没有执行,意味着两个引用 同时为空 或者同时不为空if (p == null && q == null) {return true;}// 都不为空 判断值是否一样if (p.val != q.val) {return false;}// 都不为空且值一样return isSameTree(p.left, q.left) && isSameTree(p.right, q.right);}
}

3、题三:翻转二叉树

. - 力扣(LeetCode)

3.1 思路分析

 这道题的思路很简单,就是:将每个节点的左右子树进行交换。

我们可以以前序遍历思想遍历所有节点,在访问当前树的根节点时将其左右子树进行交换。

3.2 代码

根据解题思想,很清晰的分析出时间复杂度为:O(n)

class Solution {public TreeNode invertTree(TreeNode root) {if(root == null) {return null;}//交换左右子树swapNode(root);invertTree(root.left);invertTree(root.right);return root;}public void swapNode(TreeNode root) {TreeNode tmp = root.left;root.left = root.right;root.right = tmp;}
}

4、题四:判断树是否对称

. - 力扣(LeetCode)

4.1 思路分析

要判断这颗树是对称的,

  • 就要判断,根节点的左子树和右子树是对称的 
  • 要判断根节点的左子树和右子树是对称的 ,需要左右子树的结构相同;在结构相同前提下,节点的值要相同
  • 如果结构相同了,值也相同了,那么要递归去判断:左子树的左树和右子树的右树是否轴对称(是否结构相同、值相同);以及左子树的右树和右子树的左树是否轴对称(是否结构相同、值相同)。(均满足)
  • 我们只能递归来一个节点一个节点的去遍历,去判断,去比较

 

 4.2 代码

class Solution {public boolean isSymmetric(TreeNode root) {if(root == null) {return true;}return isSymmetricChild(root.left,root.right);}public boolean isSymmetricChild(TreeNode leftTree,TreeNode rightTree) {//空树对称if(leftTree == null && rightTree == null) {return true;}//结构不相同 必定不对称if(leftTree != null && rightTree == null || leftTree == null && rightTree != null) {return false;}//节点值不相同 必定不对称if(leftTree.val != rightTree.val) {return false;}//左子树的左树和右子树的右树 //左子树的右树和右子树的左树//都要对称return isSymmetricChild(leftTree.left,rightTree.right) && isSymmetricChild(leftTree.right,rightTree.left);}
}

 5、题五:判断是否为平衡二叉树

. - 力扣(LeetCode)

5.1 思路分析

5.1.1 平衡二叉树概念

如果一棵树是平衡二叉树,那么满足:

  • 树中所有节点的左右子树高度差<=1 
  • 也就是说,每一棵子树都是平衡二叉树

根据平衡二叉树的概念,我们有以下两种思路解决问题。

5.1.2 思路一  O(n^2)

  • 递归遍历所有节点,求出每个节点的左右子树高度差,若出现>=2的情况,立即返回false
  • 当前节点的左树上的节点和右树上的节点全部满足高度差<=1 时,说明为平衡二叉树
  • 该思路时间复杂度为O(n^2),求根节点高度的方法时间复杂度本来就为O(n),而要求得所有节点的高度,时间复杂度就为O(n^2)

5.1.3 思路一代码 

class Solution {public boolean isBalanced(TreeNode root) {//空树 平衡if(root == null) {return true;}//当前节点左子树高度int h1 = getHeight(root.left);//当前节点右子树高度int h2 = getHeight(root.right);int h = Math.abs(h1-h2);//高度差//不平衡if(h >= 2) return false;//左子树和右子树同时平衡 说明整棵树平衡return isBalanced(root.left) &&isBalanced(root.right);}public int getHeight(TreeNode root) {if (root == null) {return 0;}int leftHeight = getHeight(root.left);int rightHeight = getHeight(root.right);//树的高度为 左子树高度和右子树高度的最大值+节点自身的1个高度return leftHeight > rightHeight? leftHeight + 1: rightHeight + 1;}
}

 5.1.4 思路二 :改善为O(n)【字节跳动面试题】

 思路一虽然能解决问题,但时间复杂度达到了O(n^2),并且产生了很多次重复的计算,例如:圈中节点的高度,在计算它祖先的高度时就被计算了多次,产生了很多重复的计算。

如果要将时间复杂度改善为O(n),该如何完成呢?

O(n)思路分析:

  • 我们可以只求根节点的高度
  • 在求节点高度时,我们使用递归遍历思想,返回的是左右子树高度的最大值+1(左右子树最大高度+节点自身1个高度)
  • 而我们可以将求高度的方法进行改善,在递归过程中,一旦发现左右子树高度差绝对值>=2时(说明不平衡),立即返回负数(标记,其他标记也可);如果高度差绝对值<=1时(说明平衡),返回正常的高度值即可。如若发现返回的是负数,立即一路返回负数。
  • 最终判断方法的返回值即可,若为负数,则说明该树不平衡;若为正数(高度值),说明该树平衡。

 

 5.1.5 思路二代码

class Solution {/**时间复杂度:O(n)*/public boolean isBalanced(TreeNode root) {if(root == null) {return true;}//负数说明该树不平衡return getHeight(root) > 0;}public int getHeight(TreeNode root) {if (root == null) {return 0;}int leftHeight = getHeight(root.left);//如若发现返回的是负数,则说明该树一定不平衡,一路返回负数if(leftHeight < 0) {return -1;}//如若发现返回的是负数,则说明该树一定不平衡,一路返回负数int rightHeight = getHeight(root.right);if(rightHeight < 0) {return -1;}//如果绝对值<=1 说明当前树平衡,返回高度值if(Math.abs(leftHeight - rightHeight) <= 1) {return Math.max(leftHeight , rightHeight) + 1;}else {//否则返回负数return -1;}}
}

6、题六:将二叉搜索树转化为有序的双向链表

二叉搜索树与双向链表_牛客题霸_牛客网

6.1 二叉搜索树的性质

二叉搜索树,又称二叉排序树、二叉查找树,具有以下性质:

  • 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
  • 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
  • 二叉搜索树可以为空树
  • 以中序遍历来遍历一棵二叉搜索树,那么得到的序列就是一个有序的升序序列。

6.2  思路分析

以中序遍历递归这棵二叉搜索树,将得到的根节点依次连接(修改其左右指针),得到的就是排好序的双向链表。

  • 借助中序遍历思想递归,遍历各个节点,修改指向
  • 将每个节点的left域当做双向链表中的prev域,将right域当做next域
  • 借助一个成员变量prev,存储上一个节点的地址,完成各节点指向的修改和连接

6.3 代码

时间复杂度:O(n)

public class Solution {//定义一个成员变量 存储上一个节点的地址TreeNode prev;//初始值为nullpublic void ConvertNode(TreeNode pRootOfTree) {if(pRootOfTree == null) return;//中序遍历思想递归ConvertNode(pRootOfTree.left);//修改前驱,将当前节点的left域更改为前一个节点的地址pRootOfTree.left = prev;if (prev != null) {//当prev不为空时,修改后继//将其right域修改为当前节点的地址(prev本就是当前节点的前驱)prev.right = pRootOfTree;}//更新prevprev = pRootOfTree;ConvertNode(pRootOfTree.right);}public TreeNode Convert(TreeNode pRootOfTree) {if (pRootOfTree == null) {return null;}ConvertNode(pRootOfTree);TreeNode head = pRootOfTree;//找头结点//转换成双向链表后,链表的头结点就在pRootOfTree的左边//头结点的left(前驱)为nullwhile(head.left != null) {head = head.left;}return head;}
}

7、题七:二叉树的遍历和构建

二叉树遍历_牛客题霸_牛客网

7.1  思路分析

我们知道,如果仅仅根据一个遍历方式的结果是无法构建出一棵二叉树的,必须有两个遍历方式且必须包含中序遍历。

而这道题是不同的,因为题目给出的遍历结果包含了空树,所以仅仅利用一个遍历方式我们就可以构建出二叉树。

只要构建出这棵二叉树,我们再使用中序遍历递归打印节点值就可以了。

难点在,如何使用代码根据前序遍历来构建二叉树?

思路如下:

  • 题目仅仅给出了main方法,意味着我们要自己创建节点,创建的节点类要满足树中节点的要求,包含val域(char类型)、left域、right域,同时要给出构造方法
  • 创建二叉树,肯定是要遍历前序遍历结果的字符串(这里以静态i成员遍历字符串)
  • 给出的是前序遍历结果,所以要以前序遍历的思想递归来创建树
  • 如果遍历到的字符不是'#',说明不是空树,实例出该值的节点,i++(此时仅仅实例节点,各节点间并没有连接)
  • 创建左子树,创建右子树
  • 如果遇到的字符是'#',说明遇到是空树,i++,返回空节点(递归回退),上一个节点的left或者right接收
  • 如果节点的左子树或者右子树创建完成,递归回退,上一个节点的left或者right接收
  • 如果节点的左子树和右子树都创建完成(本次递归函数结束),递归回退,上一个节点的left或者right接收
  • 也就是说,我们在递归回退的过程中,完成节点之间的连接
  • 最后一步,以中序遍历形式来访问根节点并打印

注意:这里因为题目条件限制,只能使用静态成员i来遍历字符串。但是对于我们实际的应用,不建议使用静态成员,因为当创建多个对象时,他们都会使用并改变i的值,而创建二叉树时,必须从0下标处遍历字符串!!!

7.2 代码

import java.util.Scanner;//节点
class TreeNode {TreeNode left;TreeNode right;char val;public TreeNode(char val) {this.val = val;}
}
// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {public static void main(String[] args) {Scanner in = new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 case//防止有多组测试用例,每次使用时i要置0Main.i = 0;String str = in.nextLine();TreeNode root = creatTree(str);InOrder(root);}}public static int i;public static TreeNode creatTree(String str) {TreeNode node = null;if (str.charAt(i) != '#') {//不是空树,就实例化该值的节点node = new TreeNode(str.charAt(i));i++;//在递归回退的过程中,完成节点之间的连接node.left = creatTree(str);node.right = creatTree(str);} else {//是空树,i++i++;}return node;}//中序遍历,打印根节点public static void InOrder(TreeNode root) {if (root == null) {return;}InOrder(root.left);System.out.print(root.val + " ");InOrder(root.right);}
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++中的new和模版

前言 随着C的学习&#xff0c;讲了C的发展过程、流插入、流提取、函数缺省值、类与构造等等。接下来学习C很方便的 玩意&#xff0c;函数模版。函数模版就像是模具一样&#xff0c;C会自动用模版编译出合适的函数供程序员使用。以前不同类型相同操作的函数都能通过函数模版&…

js实现数组的下标为n的对象后面新增一条对象

前言&#xff1a; js实现数组的下标为n的对象后面新增一条对象 实现方法&#xff1a; arr.splice(1, 0, obj); splice 参数1: 数组里面的第几个元素&#xff0c;你希望在第几个对象后面新增参数2: 0 表示不删除任何元素参数3: 插入的新对象 let arr [{},{},{},{}] let obj…

Android Display Graphics #从Activity看surface的创建(1)

如果图片链接失败&#xff0c;请扫码查看文章详情。 Android Display Graphics系列文章-汇总 系列文章请扫关注公众号&#xff01; 创建Activity Activity是Android系统的应用组件&#xff0c;一般情况下&#xff0c;开发者显示的内容是通过Activity展示的。比如 使用Androi…

pdf文件太大如何压缩变小一点?这几种压缩方法很实用!

pdf文件太大如何压缩变小一点&#xff1f;在日常工作与学习的海洋里&#xff0c;我们时常遭遇PDF文件的“重量级”挑战&#xff0c;这些体积庞大的文档&#xff0c;如同数字世界的巨石&#xff0c;不仅吞噬着宝贵的存储资源&#xff0c;还拖慢了处理速度&#xff0c;影响设备性…

单链表算法 - 链表的回文结构

链表的回文结构_牛客题霸_牛客网对于一个链表&#xff0c;请设计一个时间复杂度为O(n),额外空间复杂度为O(1)的算法&#xff0c;判断其是否为。题目来自【牛客题霸】https://www.nowcoder.com/practice/d281619e4b3e4a60a2cc66ea32855bfa思路1: /* struct ListNode {int val;st…

.net dataexcel 脚本公式 函数源码

示例如: ScriptExec(""sum(1, 2, 3, 4)"") 结果等于10 using Feng.Excel.Builder; using Feng.Excel.Collections; using Feng.Excel.Interfaces; using Feng.Script.CBEexpress; using Feng.Script.Method; using System; using System.Collections.Gen…

MATLAB R2023b下载安装教程汉化中文版设置

MATLAB R2023b下载安装教程汉化中文版设置 Matlab 是一款功能强大的商业数学软件 Matlab&#xff08;Matrix Labortory&#xff09;即矩阵实验室&#xff0c;它在数值计算、数据分析、算法开发、建模与仿真等众多领域都发挥着重要作用。 Matlab 具有以下显著特点和优势&…

TYPE-C接口PD取电快充协议芯片ECP5701:支持PD 2.0和PD 3.0(5V,9V,12V,15V,20V)

随着智能设备的普及&#xff0c;快充技术成为了越来越多用户的刚需。而TYPE-C接口作为新一代的USB接口&#xff0c;具有正反插、传输速度快、充电体验好等优点&#xff0c;已经成为了快充技术的主要接口形式。而TYPE-C接口的PD&#xff08;Power Delivery&#xff09;取电快充协…

MySQL学习记录 —— 이십삼 MySQL服务器文件系统(3)

文章目录 1、数据字典2、系统表各种系统表 Mysql Schema是⼀个系统库&#xff0c;表中存储了MySQL服务器运行时所需的信息。广义上&#xff0c;mysql schema包含存储MySQL程序基本数据的数据字典和用于其他操作目的的系统表。数据字典表和系统表位于数据目录下一个名为mysql.ib…

角点检测及MATLAB实现

一、角点简介 角点通常指的是两条直线构成角时的交点。‌在更广泛的应用中&#xff0c;‌角点这一概念也被扩展到数字图像处理领域&#xff0c;‌其中角点被定义为图像中物体轮廓线的连接点&#xff0c;‌这些点在某方面属性特别突出&#xff0c;‌即在某些属性上强度最大或者最…

为什么不要碰自媒体

要是失业了&#xff0c;搞自媒体&#xff0c;可行吗&#xff1f;毫无希望&#xff01; 如今的自媒体早卷得不成样子了&#xff0c;很难再有机会&#xff0c;根本原因在于几乎没有增量用户的同时&#xff0c;存量用户也不再有剩余时间&#xff0c;全量用户的时间早已被几个自媒…

空间计算开发:Volu的集成开发工具包

在空间计算技术迅速发展的今天,VR和AR项目的开发需求日益增长。Volu,一个面向空间计算赛道的开发者工具,正致力于简化这一过程。本文将深入探讨Volu如何通过其集成环境,为开发者提供一站式的解决方案。 一、定位:空间计算的得力助手 Volu定位为一个专为空间开发设计的集…

verilog基础语法入门

文章目录 前言一、模块定义1. 模块声明2. 端口定义3. 信号类型声明4. 逻辑功能定义 二、运算符与表达式1. 算术运算符2. 逻辑运算符3. 位运算符4. 关系运算符5. 等式运算符6. 缩减运算符7. 移位运算符8. 条件运算符9. 位拼接运算符 三、语句1. 赋值语句2. 块语句3. 条件语句4. …

HarmonyOS NEXT学习——@Styles、@Extend、stateStyles

Styles装饰器 定义组件重用样式 仅支持通用属性和通用事件不支持参数可以定义全局和组件内使用&#xff0c;全局使用需要加function // 全局 Styles function functionName() { ... }// 在组件内 Component struct FancyUse {Styles fancy() {.height(100)} }组件内Styles的优…

C# 之工控机数据类型 高低位(大小端)、BitConverter、IsLittleEndian、字节数组转换(高低位)

八种基本数据类型&#xff1a;byte、short、int、long、float、double、boolean、char byte 8位、有符号的以二进制补码表示的整数 min : -128&#xff08;-2^7&#xff09; max: 127(2^7-1) default: 0 对应包装类&#xff1a;Byte short 16位、有符号的以二进制补码表示…

keil中GD32 MCU IAP中APP的存储地址如何设置?

前面和大家聊过什么是IAP&#xff0c;那么IAP中APP的存储地址该如何设置呢&#xff1f; 以keil为例&#xff0c;打开工程的option选项卡&#xff1a; 将IROM1中的地址改为你想要保存的位置&#xff0c;比如0x08008000开始的位置&#xff1a; 这样通过keil烧录&#xff0c;程序…

光伏电站设计的关键部分

光伏设计&#xff0c;作为光伏发电项目成功实施的前提与基础&#xff0c;其科学性与合理性直接关乎项目的发电效率、经济性及环境适应性。本文将从选址规划、系统设计、组件选型、电气布局及运维考虑等几个方面&#xff0c;探讨光伏设计的关键部分。 一、选址踏勘 光伏项目踏勘…

钡铼Profinet、EtherCAT、Modbus、MQTT、Ethernet/IP、OPC UA分布式IO系统BL20X系列耦合器

BL20X系列耦合器是钡铼技术开发的一款用于分布式I/O系统的设备&#xff0c;专为工业环境下的高速数据传输和远程设备控制而设计&#xff0c;支持多种工业以太网协议&#xff0c;包括Profinet、EtherCAT、Modbus、MQTT、Ethernet/IP和OPC UA等。如果您正在考虑部署BL20X系列耦合…

c++人脸识别项目,满足工业界对于人脸识别系统的高标准需求!(一)

目录 引言&#xff1a; 环境构建&#xff08;本文使用cmake&#xff0c;开发环境ubuntu22.04&#xff0c;IDE为clion&#xff09; 项目文件构造 CMakeLists.txt编写 简单头文件 最初成员函数实现 add函数实现思路 search函数实现思路 main函数简单实现 思路 添加环…

Spring Boot集成qwen:0.5b实现对话功能

1.什么是qwen:0.5b&#xff1f; 模型介绍&#xff1a; Qwen1.5是阿里云推出的一系列大型语言模型。 Qwen是阿里云推出的一系列基于Transformer的大型语言模型&#xff0c;在大量数据&#xff08;包括网页文本、书籍、代码等&#xff09;进行了预训练。 硬件要求&#xff1a;…