LeetCode刷题之HOT100之二叉树的最近公共祖先

2024 7/1 新的一个月来啦!也算是迎来了暑假,可惜我们没有暑假,只能待实验室,中途会有10天小假。Anyway,做题啦

1、题目描述

在这里插入图片描述

2、算法分析

又来到了树的部分,要找最近的公共祖先。想到树就会想到DFSBFS
祖先的定义: 若节点 p 在节点 root 的左(右)子树中,或 p=root ,则称 rootp 的祖先。这与人类社会关系唯一不同的是,自己也可以是自己的祖先。
最近公共祖先的定义: 设节点 root 为节点 p,q 的某公共祖先,若其左子节点 root.left 和右子节点 root.right 都不是 p,q 的公共祖先,则称 root 是 “最近的公共祖先” 。
这里使用递归方式,利用DFS算法。那么符合条件的最近公共祖先 一定满足如下条件:

((leftSon && rightSon) || ((root.val == p.val || root.val == q.val) && (leftSon || rightSon)))

如果 pq 分别在左右子树中,或者 pq 中的一个等于当前节点并且 pq 在子树中,则当前节点为最近公共祖先

3、代码

 // 定义一个私有成员变量res,用于存储找到的最低公共祖先节点
    private TreeNode res;
    // 构造函数,初始化res为null
    public Solution(){
        this.res = null;
    }
    // 定义一个私有递归方法dfs,用于在二叉树中查找最低公共祖先  
    // 参数:  
    //   root - 当前遍历的节点  
    //   p - 给定的节点p  
    //   q - 给定的节点q  
    // 返回值:  
    //   一个布尔值,表示p和q是否都在当前子树中
    private boolean dfs(TreeNode root, TreeNode p, TreeNode q){
        // 如果当前节点为空,说明已经遍历到叶子节点的下方,返回false 
        if(root == null ){
            return false;
        }
        // 递归地在左子树中查找p和q 
        boolean leftSon = dfs(root.left, p, q);
        // 递归地在右子树中查找p和q
        boolean rightSon = dfs(root.right, p, q);
        // 如果 p 和 q 分别在左右子树中,或者 p 和 q 中的一个等于当前节点并且 p 和 q 在子树中,则当前节点为最近公共祖先
        if((leftSon && rightSon) || ((root.val == p.val || root.val == q.val) && (leftSon || rightSon))){
            res = root;
        }
        // 返回p和q是否在当前子树中(包括当前节点)
        return leftSon || rightSon || (root.val == p.val || root.val == q.val);
    }
    // 公开方法,用于调用私有方法dfs并返回找到的最低公共祖先节点 
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
        // 调用dfs方法
        this.dfs(root, p, q);
        // 返回找到的最低公共祖先节点
        return this.res;   
    }

我们举一个例子来说明:如下图所示,我们要找7和8的最近公共祖先。
在这里插入图片描述
以下是调用lowestCommonAncestor方法并找到LCA的步骤:

  • 从根节点3开始调用dfs方法,并将78作为要查找的节点。
  • dfs方法首先检查当前节点3,然后递归地在其左子树(以5为根)中查找78
  • 在左子树中,dfs方法会首先检查节点5,然后递归地在节点5的左子树(以6为根)中查找78。由于节点6没有子节点,所以dfs(6, 7,8)返回false,表示78都不在节点6的子树中。
  • 接着,dfs方法会检查节点5的右子树(以2为根)。在右子树中,它发现节点72的右子节点,所以dfs(2, 7,8)返回true,因为节点7在节点2的子树中。
  • 此时,dfs方法回到节点5,并发现leftSonfalse(因为7不在左子树中),而rightSontrue(因为7在右子树中)。但是,因为78没有同时在左右子树中,所以节点5不是78LCA
  • dfs方法返回true到根节点3,表示7在节点3的左子树中。
  • 接下来,dfs方法检查根节点3的右子树(以1为根)。在右子树中,它发现节点81的右子节点,所以dfs(1, 7, 8)返回true,因为节点8在节点1的子树中。
  • 此时,dfs方法再次回到根节点3,并发现leftSonrightSon都为true(因为7在左子树中,8在右子树中)。根据条件(leftSon&& rightSon),节点3被确定为78LCA,并将res设置为节点3
  • 最后,lowestCommonAncestor方法返回存储在res中的节点,即节点3
    因此,节点78LCA是节点3

4、复杂度分析

  • 时间复杂度:O(N),其中 N 是二叉树的节点数。二叉树的所有节点有且只会被访问一次,因此时间复杂度为 O(N)。
  • 空间复杂度:O(N) ,其中 N 是二叉树的节点数。递归调用的栈深度取决于二叉树的高度,二叉树最坏情况下为一条链,此时高度为 N,因此空间复杂度为 O(N)。

ok啦,做完啦,拜拜!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

李一桐遭遇蜈蚣惊魂

李一桐遭遇“蜈蚣惊魂”!刘宇宁展现真男人本色在娱乐圈的幕后,总有一些心跳加速的惊险。近日,李一桐在拍戏时遭遇了一场“蜈蚣惊魂”,让无数粉丝和网友为她捏了一把冷汗。而在这场惊险的遭遇中,刘宇宁展现出了真男人的…

【Spring Boot】spring boot环境搭建

1、环境准备 JDK安装:确保安装了Java Development Kit (JDK) 1.8或更高版本。JDK是Java编程的基础,Spring Boot项目需要它来编译和运行。Maven或Gradle安装:选择并安装Maven或Gradle作为项目构建工具。Maven通过pom.xml文件来管理项目的依赖…

深入浅出:npm 常用命令详解与实践

在现代的前端开发流程中,npm(Node Package Manager)已经成为了不可或缺的一部分。它不仅帮助我们有效地管理项目中的依赖包,还提供了一系列强大的命令来优化开发体验。在这篇博客中,我们将深入探讨 npm 的常用命令&…

【正点原子K210连载】 第十五章 按键中断实验 摘自【正点原子】DNK210使用指南-CanMV版指南

1)实验平台:正点原子ATK-DNK210开发板 2)平台购买地址https://detail.tmall.com/item.htm?id731866264428 3)全套实验源码手册视频下载地址: http://www.openedv.com/docs/boards/xiaoxitongban 第十五章 按键中断实…

问题解决|endnote文献手工导入

一、背景介绍 手工导入一篇文献是指手动编辑文献的相关信息Preference。为什么要手动这么麻烦?因为有的文献比较老只有纸质版本,有的文献信息不全,有的则是没有编码无法识别等等,需要手工录入;一般需要手工录入的情况比…

Decorators与类

在Python中,装饰器(decorator)是一种用于修改函数或方法行为的特殊函数。装饰器可以用于函数、方法和类。在类中使用装饰器可以增强类的方法、属性,甚至整个类的功能。以下是一些关于我对装饰器与类的详细信息和示例教程。 1、问题…

计算机系统导论

第一章 计算机系统基本概述 【1】世界上第一台计算机 1946 年由美国宾夕法尼亚大学研制出世界上第一台电子数字计算机,取名 ENIAC。由此 诞生了“第一个电子的大脑” 【2】计算机的发展阶段 第一个发展阶段:1946-1956 年电子管计算机的时代.1946 年…

Halcon 特征检测使用

一 Region area: 面积row: 中心的行坐标column: 中心的列坐标width: 区域的宽度(平行于坐标轴)height: 区域的高度(平行于坐标轴)row1: 左上角的行坐标column1: 左上角的列坐标row2: 右下角的行坐标column2: 右下角的列坐标‘ra’; 椭圆的长半轴…

IMU用于水下机械臂遥操作

在当今科技飞速发展的时代,探索深海奥秘与执行水下任务如今有了新帮手——一款能模拟人类手臂动作的水下机械臂。这款由波兰科学家携手机器人公司联手打造的创新产品能够精确复现人类手臂的动作,其精髓在于构建了一个由惯性测量单元(IMU&…

【技巧】ArcGIS Pro设置自动保存数据编辑内容

一、工程文件自动保存 ArcGIS Pro软件的工程也可以自动保存备份。默认备份时间是5分钟,您可以在【工程】→【选项】→【常规】→【工程恢复】中调整自动备份时间。 二、数据编辑自动保存 操作方法:【工程】→【选项】→【编辑】→【会话】,勾…

fastapi+vue3前后端分离开发第一个案例整理

开发思路 1、使用fastapi开发第一个后端接口 2、使用fastapi解决cors跨域的问题。cors跨域是浏览器的问题,只要使用浏览器,不同IP或者不同端口之间通信,就会存在这个问题。前后端分离是两个服务,端口不一样,所以必须要…

flex讲解

随着前端技术的不断发展和更新,flex布局成为前端布局的主流。但是仍然有很多前端新手搞不懂flex到底怎么用!!!今天我们就来好好讲讲flex布局 老规矩先上定义 什么是flex布局 布局的传统解决方案,基于盒状模型&#x…

DNF手游鬼剑士攻略:全面解析流光星陨刀的获取与升级!云手机强力辅助!

《地下城与勇士》(DNF)手游是一款广受欢迎的多人在线角色扮演游戏,其中鬼剑士作为一个经典职业,因其强大的输出能力和炫酷的技能特效,吸引了众多玩家的青睐。在这篇攻略中,我们将详细介绍鬼剑士的一把重要武…

智慧路灯可视化:点亮城市管理的新篇章

智慧路灯可视化系统通过图扑 HT 实时数据采集和分析,将城市每一盏路灯的状态、能耗和故障信息一目了然地展示在管理平台上。高效的监控与管理不仅提升了公共照明的维护效率,减少人工巡检成本,还支持节能策略,实现智慧城市的可持续…

国际短信API的功能有哪些?如何配置使用?

国际短信API的合规性如何保障?国际短信API使用教程? 国际短信API不仅仅是一个发送短信的工具,它还包含了许多强大的功能,能够帮助企业更好地管理和优化他们的通信策略。AoKSend将详细探讨国际短信API的主要功能。 国际短信API&a…

Excel表格转Tex工具推荐

为了制作符合 SCI 论文要求的表格,直接用 LaTeX 编写通常比较复杂。我们可以先在 Excel 中绘制好所需的表格(最好加上边框)。最近我发现了一个非常好用的 Excel 转 LaTeX 工具,能够让 LaTeX 表格的编写变得非常方便。 工具&#…

跨境电商内卷时代,亚马逊卖家如何低成本提升产品曝光与销量?

在跨境电商领域,随着市场的日益饱和和竞争的加剧,卖家们普遍面临着一个共同的挑战:流量稀缺,转化率低。为了在这个“内卷”严重的环境中脱颖而出,许多卖家不惜投入大量资金和资源,尝试各种站内和站外推广手…

20240701 每日AI必读资讯

🏫AI真炼丹:整整14天,无需人类参与 - 英矽智能推出全球首个AI参与决策的生物学实验室,实现了14天内完成靶点发现和验证的全自动化闭环实验。 - 该实验室由PandaOmics平台驱动,集成多种预测模型和海量数据&#xff0…

前端:多服务端接口资源整合与zip打包下载

项目需求 前端项目开发中,有一个页面需要去整合多个服务接口返回的数据资源,并且需要将这多个服务接口接口返回的数据进行资源压缩,最终打包成zip压缩包,并在客户端完成下载。 基本需求梳理如下, 实现思路 这个需求点其实本质上还是传统的“文件下载”功能需求,常见的例如…

Golang基础问题

Go基础 文章目录 Go基础● Go有那些关键字?● Go方法与函数的区别?● Go函数返回局部变量的指针是否安全?● Go函数参数传递是值传递还是引用传递?● defer关键字的实现原理?● 内置函数make和new的区别?●…