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

news2025/1/24 17:46:41

👉 ​​​​​​目录 👈

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;
        }
        //走到这里,说明两棵树的结构相同且不为空,那判断他们的节点值即可
        //若节点值不相同,则说明两棵树不相同,返回false
        if(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) {
        //如果递归到空,说明没有找到子树,返回false
        if(root == null) {
            return false;
        }
        //判断当前节点的整颗树和子树是否相同即可
        if(isSameTree(root,subRoot)) {
            return true;
        }
        //如果递归遇见了子树 则一路返回true
        if(isSubtree(root.left,subRoot)) return true;
        if(isSubtree(root.right,subRoot)) return true;
        //本次递归没有找见子树 返回false
        return 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;//初始值为null

    public void ConvertNode(TreeNode pRootOfTree) {
        if(pRootOfTree == null) return;
        //中序遍历思想递归
        ConvertNode(pRootOfTree.left);
        
        //修改前驱,将当前节点的left域更改为前一个节点的地址
        pRootOfTree.left = prev;
        if (prev != null) {
            //当prev不为空时,修改后继
            //将其right域修改为当前节点的地址(prev本就是当前节点的前驱)
            prev.right = pRootOfTree;
        }
        //更新prev
        prev = pRootOfTree;

        ConvertNode(pRootOfTree.right);
    }
    public TreeNode Convert(TreeNode pRootOfTree) {
        if (pRootOfTree == null) {
            return null;
        }
        ConvertNode(pRootOfTree);

        TreeNode head = pRootOfTree;
        //找头结点
        //转换成双向链表后,链表的头结点就在pRootOfTree的左边
        //头结点的left(前驱)为null
        while(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要置0
            Main.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.coloradmin.cn/o/1935816.html

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

相关文章

小模型大突破!神经网络透视空间异质性,准确描述复杂地理现象

为推进 AI4S 的普适化&#xff0c;降低学术机构科研成果的传播壁垒&#xff0c;为更多行业学者、科技爱好者及产业单位提供交流平台&#xff0c;HyperAI超神经策划了「Meet AI4S」系列直播栏目&#xff0c; 邀请深耕 AI for Science 领域的科研人员或相关单位&#xff0c;以视频…

新时代多目标优化【数学建模】领域的极致探索——数学规划模型

目录 例1 1.问题重述 2.基本模型 变量定义&#xff1a; 目标函数&#xff1a; 约束条件&#xff1a; 3.模型分析与假设 4.模型求解 5.LINGO代码实现 6.结果解释 ​编辑 7.敏感性分析 8.结果解释 例2 奶制品的销售计划 1.问题重述 ​编辑 2.基本模型 3.模…

北京邮电大学,中央空调的分户计费系统

北京邮电大学 中央空调如何公平、公正、合理的收取费用&#xff0c;一直都是各建筑管理者的首要问题。北京邮电大学也面临着能源分配不公&#xff0c;学校管理者空调收费管理困难等问题。根据学校的具体情况&#xff0c;拓森为其制定了一套中央空调管理运营方案—无线中央空调…

jupyter学习笔记

简介 Jupyter Notebook是一个Web应用程序&#xff0c;它可以在网页页面中直接编写代码和运行代码&#xff0c;代码的运行结果也会直接在代码块下显示。 安装使用 前提&#xff1a;必须安装python 先升级pip至最新版本 pip3 install --upgrade pip安装jupyter notebook pi…

springboot的简单应用

Mvc与三层架构 创建Spring项目 勾选web和mabais框架 配置yml文件&#xff0c;这里创建spring项目默认生成的是propertise文件&#xff0c;但是properties文件的格式看起来没有yml文件一目了然。yml文件配置数据库还有映射mapper层的xml文件以及设置日志级别&#xff0c;比如map…

刚起步的家庭海外仓:涉及到的全部业务优化流程

对于家庭海外仓来说&#xff0c;最难的阶段应该就是刚起步的时候。对业务流程不熟悉&#xff0c;也没有客户积累&#xff0c;本身的预算又十分有限。 在这个情况下应该注意什么&#xff0c;怎样才能顺利的开展业务&#xff1f;今天我们就针对这个问题详细的梳理了一下家庭海外…

尚品汇-(二十一)

目录&#xff1a; &#xff08;1&#xff09;使用redis实现分布式锁 &#xff08;2&#xff09;优化之设置锁的过期时间 &#xff08;3.&#xff09;优化之UUID防误删 &#xff08;4&#xff09;优化之LUA脚本保证删除的原子性 &#xff08;1&#xff09;使用redis实现分布…

go语言编程 小试牛刀 goroutine和reflect知识点

&#xff08;一&#xff09;goroutine package _caseimport "fmt"// sum 函数计算整数切片 values 的总和&#xff0c;并将结果发送到 resultChan 通道中 func sum(values []int, resultChan chan int) {sum : 0for _, value : range values {sum value}resultChan…

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

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

智能一体式闸门在灌区中的应用

在现代化的农业灌溉领域&#xff0c;智能一体式闸门作为一种集自动化、智能化、高效能于一体的先进设备&#xff0c;正逐渐在灌区管理中发挥着重要作用。 灌区是农业生产的重要基地&#xff0c;其水资源的管理和利用直接关系到农作物的生长和产量。然而&#xff0c;传统的闸门管…

旋转中的图片视觉差效果

Hello&#xff0c;亲爱的宝子们&#xff1f;最近我一个前端架构师却临时顶替产品经理的工作&#xff0c;导致最近一周实在太忙了&#xff0c;都没有来得及更新文章。在这里想大家道歉了&#xff01;也想厚颜无耻的问问大家想我了吗&#xff1f;(●◡●) 今天给大家带来一个非常…

Vue 使用 Element UI 组件库

https://andi.cn/page/621589.html

RAG介绍

一&#xff0c;RAG概述 RAG&#xff08;Retrieval-Augmented Generation&#xff0c;检索增强生成&#xff09;是一种结合了信息检索与生成任务的技术&#xff0c;它通过将外部知识库与大模型的生成能力相结合&#xff0c;提高了生成内容的准确性和丰富性。以下是关于RAG应用的…

【产品那些事】固件安全-关于OTA升级包分析

文章目录 前言什么是OTA?升级包(固件)的类型和架构案例tp-link路由器升级包怎么解包分析?binwalk安装及使用ubi_reader安装及使用unsquashfs安装及使用某车企OTA升级包通用Android OTA解包相关分区第二层解包前言 什么是OTA? OTA(Over-the-Air)是一种通过无线通信网络(…

go的Mutex实现原理及演进

下面的这个是对于昨天晚上读的几篇关于go中锁的文章知识点的总结 文章目录 1. 引言1.1 并发编程的挑战1.2 Mutex 的角色1.3 Mutex 设计的演进1.4 技术追求的美妙 引言部分详细解释引言部分注意点引言部分流程图 2. Mutex 架构演进2.1 初版 Mutex 设计2.2 性能优化 - 给新人机会…

【ffmpeg】一篇文章搞定YUV

文章目录 前言YUV是什么&#xff1f;YUV的用途YUV采样格式采样格式是什么YUV采样格式有哪些YUV采样格式的区别 YUV与RGBRGB 颜色空间YUV 颜色空间RGB 与 YUV 的比较RGB 转 YUV 公式YUV 转 RGB 公式注意事项 YVU数据计算通用公式4:4:4 采样格式4:2:2 采样格式4:2:0 采样格式实例…

win10删除鼠标右键选项

鼠标右键菜单时&#xff0c;发现里面的选项特别多&#xff0c;找一下属性&#xff0c;半天找不到。删除一些不常用的选项&#xff0c;让右键菜单变得干净整洁。 1、按下键盘上的“winR”组合按键&#xff0c;调出“运行”对话框&#xff0c;输入“regedit”命令&#xff0c;点击…

达梦数据库的系统视图v$rapply_log_task

达梦数据库的系统视图v$rapply_log_task 达梦数据库的V$RAPPLY_LOG_TASK视图是一个动态性能视图&#xff0c;主要用于在备库上查询。该视图需要在备库上查询&#xff08;DMDSC 备库需要在控制节点&#xff08;重演节点&#xff09;上查询&#xff09;&#xff0c;用于查询备库…

专业PDF编辑工具:Acrobat Pro DC 2024.002.20933绿色版,提升你的工作效率!

软件介绍 Adobe Acrobat Pro DC 2024绿色便携版是一款功能强大的PDF编辑和转换软件&#xff0c;由Adobe公司推出。它是Acrobat XI系列的后续产品&#xff0c;提供了全新的用户界面和增强功能。用户可以借助这款软件将纸质文件转换为可编辑的电子文件&#xff0c;便于传输、签署…

RocketMQ单结点安装/Dashboard安装

目录 1.安装NameServer 2.安装Broker 3.使用自带工具测试数据发送 4.使用DashBoard进行查看 5.关闭相关设备 前置条件&#xff1a;两台虚拟机CentOS Linux release 7.5.1804(ps:当然也可以都部署在一台机器上) RocketMq属于天生集群。需要同时启动nameServer和Broker进行…