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

news2025/1/15 17:34:46

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啦,做完啦,拜拜!

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

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

相关文章

护眼灯哪些牌子好?几款最好的护眼灯品牌排行榜分享

在当代社会,随着工作压力和学业负担的增加,人们的用眼时间越来越长,因此保护眼睛的需求变得愈发迫切。护眼台灯作为一种护眼产品,已经逐渐普及,成为许多人的助手。然而护眼灯哪些牌子好?今天,我…

3D交互可视化编辑器求推荐,最好是针对企业级使用的?

企业级使用的3D交互可视化编辑器,支持编辑和调整2D、3D渲染及交互设置,以下几款可以关注了解一下: 1、Unity:一个广泛使用的跨平台游戏引擎,由Unity Technologies开发。支持开发者创建2D和3D游戏、交互式应用以及虚拟…

4个文章生成器免费版分享,让文章创作更轻松便捷

在当今这个信息飞速传播的时代,文章创作的重要性愈发凸显。无论是从事内容创作的专业人士,还是偶尔需要撰写文章的普通大众,都希望能更高效地完成文章创作任务。而在实际操作中,我们常常会遇到思路卡顿、没有创作灵感的问题。今天…

全网最详细的 gin框架请求数据绑定Bind 源码解析 -- 帮助你全面了解gin框架的请求数据绑定原理和方法

在gin框架中,我们可以将多种请求数据(json, form,uri,header等)直接绑定到我们定义的结构体,底层是通过反射方式获取我们定义在结构体上面的tag来实现请求数据到我们的结构体数据的绑定的。 在gin的底层有2大体系的数据…

Apple苹果可能会在今年秋天宣布与Google Gemini谷歌双子座的交易

如果你对迄今为止唯一能与苹果设备集成的人工智能模型是ChatGPT感到失望,听起来你不必等待很长时间就能改变这种情况。据彭博社(Bloomberg)的马克古尔曼(Mark Gurman)今天在他的Power On时事通讯中称,苹果将…

GuLi商城-商品服务-API-三级分类-删除-逻辑删除

注意:官方文档说logic配置可以省略,代码中直观些,配上吧 逻辑删除注解: 实体类字段上加逻辑删除注解: 启动nacos: 启动商品服务: postman测试: 数据库字段值改成了0,说明…

Decorators与类

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

vivado VIO IP核

参考:pg159 VIO:可以模拟输入/输出功能,实时监视和修改FPGA中的信号,用于调试和验证,与ILA相比,VIO无需占用RAM资源。 VIO IP的输出对于FPGA内部逻辑是输入信号,可以在调试界面设置输入值&…

第6章 复制

文章目录 前言1.配置1.1建立复制1.2断开复制1.3 安全性1.4 只读1.5 传输延迟 2. 拓扑2.1.一主一从结构2.2.一主多从结构2.3.树状主从结构 3.原理3.1复制过程3.2数据同步3.3全量复制 前言 复制功能,实现了相同数据的多个Redis副本。复制功能是高可用Redis的基础&…

STM32第十二课:ADC检测烟雾浓度(MQ2)

文章目录 需求一、MQ-2 气体传感器特点应用电路及引脚 二、实现流程1.开时钟,分频,配IO2.配置ADC的工作模式3.配置通道4.复位,AD校准5.数值的获取 需求实现总结 需求 使用ADC将MQ2模块检测到的烟雾浓度模拟量转化为数字量。 最后&#xff0c…

[Go 微服务] Kratos 验证码业务

文章目录 1.环境准备2.验证码服务2.1 kratos 初始化验证码服务项目2.2 使用 Protobuf 定义验证码生成接口2.3 业务逻辑代码实现 1.环境准备 protoc和protoc-gen-go插件安装和kratos工具安装 protoc下载 下载二进制文件:https://github.com/protocolbuffers/protobu…

CocosCreator构建IOS教程

CocosCreator构建IOS教程 添加include: Header Search Paths:拖拽include过来 添加SoundEngine: Header Search Paths: 把SoundEngine POSIX Common 三个文件夹拖拽到里面去

IEEE TNNLS | 脑电(EEG)自监督学习

摘要 数十年的研究表明,与传统的统计技术相比,机器学习在探索脑电图(EEG)记录中嵌入的高度非线性模式方面具有优势。然而,即使是最先进的机器学习技术也需要相对较大且标记完整的EEG存储库。EEG数据的收集和标记成本高昂。此外,由…

Google ghOSt 调度器分析(4)

调度器的优缺点 *ghOSt* 调度器的优缺点优点缺点*ghost* 与 *CFS* 调度运行时间比较 ghOSt 调度器的优缺点 优点 逻辑简单,实现简单;它只是在内核中增加了两个调度类,通过对这两个调度类的操作来完成相应任务的优先级的提升等操作&#xff…

HarmonyOS(42) Divider 分割器组件 实现分割线

Divider分割线 简介使用示例参考目录 简介 该组件可以帮助我们实现 水平分割线和竖直分割线,同时支持设置分割线的宽度、颜色、和两端的样式 使用示例 横向分割线 ,默认就是横向分割 // Horizontal dividerColumn() {this.Block()Divider()this.Bloc…

如何使用ChatGPT提高数学建模竞赛的获奖概率

如何使用ChatGPT提高数学建模竞赛的获奖概率 数学建模助手GPT https://chatgpt-plus.top/g/g-OX0D7uMn9-shu-ju-jian-mo-zhu-shou-by-maynor 1. 问题分析与理解 在数学建模的初期,准确理解问题的背景和要求至关重要。通过使用ChatGPT,你可以&#xff…

【基础篇】第4章 查询与过滤

在Elasticsearch的世界里,高效地从海量数据中检索出所需信息是其核心价值所在。本章将深入解析查询与过滤的机制,从基础查询到复合查询,再到全文搜索与分析器的定制,为你揭开数据检索的神秘面纱。 4.1 基本查询 4.1.1 Match查询…

Spring MVC 获取三个域(request请求域,session 会话域,application 应用域)对象的方式

1. Spring MVC 获取三个域(request请求域,session 会话域,application 应用域)对象的方式 文章目录 1. Spring MVC 获取三个域(request请求域,session 会话域,application 应用域)对象的方式2. Servlet中的三个域对象3. 准备工作3…

8.12 矢量图层面要素单一符号使用十二(插值线渲染边界)

文章目录 前言插值线渲染边界(Outline: Interpolated Line)QGis设置面符号为插值线渲染边界(Outline: Interpolated Line)二次开发代码实现插值线渲染边界(Outline: Interpolated Line) 总结 前言 本章介绍…

【Linux】部署NFS服务实现数据共享

👨‍🎓博主简介 🏅CSDN博客专家   🏅云计算领域优质创作者   🏅华为云开发者社区专家博主   🏅阿里云开发者社区专家博主 💊交流社区:运维交流社区 欢迎大家的加入&#xff01…