【LeetCode经典题解】搞定二叉树最近公共祖先:递归法+栈存路径法,附代码实现

【LeetCode经典题解】搞定二叉树最近公共祖先:递归法+栈存路径法,附代码实现
在这里插入图片描述
🎁个人主页:User_芊芊君子
🎉欢迎大家点赞👍评论📝收藏⭐文章
🔍系列专栏:Java.数据结构
在这里插入图片描述


在这里插入图片描述


【前言】

二叉树的最近公共祖先是数据结构中的经典问题,无论是算法面试还是实际开发都高频出现。本文将从问题本身出发,拆解递归法与栈存路径法两种核心思路的逻辑步骤,并附上完整代码实现,帮你快速掌握这一考点。

文章目录:

一、二叉树的最近共同祖先

在这里插入图片描述

二、思路分析

方法一:递归法

判空

  • 如果根节点为空,直接返回null
  • 如果当前节点是p或者q,那么这个节点就是最近公共祖先

递归

  • 分别遍历当前节点的左子树和右子树,找到p和q的最近公共祖先,存于TreeNode leftRet或者TreeNode rightRet

结果

  • 如果左右子树都找到,那么当前节点就是p和q的最近公共祖先
  • 如果左子树找到,那就返回左子树的结果;

如果右子树找到,那就返回右子树的结果;

在这里插入图片描述

方法二:栈存路径法

获取路径

通过深度优先搜索,分别找到从根节点到p和q的路径中所有节点,并分别存入两个栈中

对齐栈长度

比较两个栈中元素长度,将较长的栈中元素弹出,然后比较两个栈中的下一深度元素是否相等

找公共祖先

同时弹出两个栈中不相等的栈顶元素,第一个相同的节点就是最近公共祖先

在这里插入图片描述

三、代码展示

方法一:递归法

publicTreeNodelowestCommonAncestor(TreeNode root,TreeNode p,TreeNode q){//当前节点为空,直接返回nullif(root ==null){returnnull;}//当前节点是p或者q,返回当前节点(祖先)if(root == p || root == q){return root;}//递归探索左子树,找到p和q的最近祖先TreeNode leftRet =lowestCommonAncestor(root.left, p, q);//递归探索右子树,找到p和q的最近祖先TreeNode rightRet =lowestCommonAncestor(root.right, p, q);//左子树和右子树都找到if(leftRet !=null&& rightRet !=null){return root;//左子树找到}elseif(leftRet !=null){return leftRet;}else{//右子树找到return rightRet;}}

方法二:栈存路径法

publicbooleangetPath(TreeNode root,TreeNode node,Stack<TreeNode> stack){if(root ==null){returnfalse;}//将当前节点压入路径 stack.push(root);//找到目标节点if(root == node){returntrue;}//递归搜索左子树boolean flg =getPath(root.left,node,stack);if(flg){returntrue;}//递归搜索右子树 flg =getPath(root.right,node,stack);if(flg){returntrue;}//左右子树都没找到,弹出当前节点 stack.pop();returnfalse;}publicTreeNodelowestCommonAncestor(TreeNode root,TreeNode p,TreeNode q){if(root ==null){returnnull;}//找到跟到p和q的路径Stack<TreeNode> stackp =newStack<>();Stack<TreeNode> stackq =newStack<>();getPath(root,p,stackp);getPath(root,q,stackq);//对齐两个栈长度int sizep = stackp.size();int sizeq = stackq.size();int size = sizep-sizeq;if(size>0){//stacp更长,弹出多余元素while(size !=0){ stackp.pop(); size--;}}else{////stacq更长,弹出多余元素 size = sizeq-sizep;while(size !=0){ stackq.pop(); size--;}}//此时两个栈大小一样,//同时弹出栈顶,找到相同的while(!stackp.isEmpty()&&!stackq.isEmpty()){if(stackp.peek().equals(stackq.peek())){return stackp.peek();} stackp.pop(); stackq.pop();}returnnull;}

四、总结

通过递归法的“自底向上”回溯,以及栈存路径法的“路径对比”思路,我们可以高效求解二叉树的最近公共祖先问题——前者利用递归特性简化代码,后者通过显式路径存储更直观易懂。结合本文的思路分析与代码示例,你可以根据场景灵活选择解法,轻松应对这类二叉树算法题。

Read more

Python百度搜索API终极指南:5分钟掌握免密钥搜索技术

🎯 为什么你需要这个搜索神器? 【免费下载链接】python-baidusearch自己手写的百度搜索接口的封装,pip安装,支持命令行执行。Baidu Search unofficial API for Python with no external dependencies 项目地址: https://gitcode.com/gh_mirrors/py/python-baidusearch 在日常开发中,你是否经常遇到这些问题: * 数据获取困难:手动复制粘贴搜索结果,耗时耗力 * 代码集成复杂:想要在应用中集成搜索功能,却苦于没有合适的API * 命令行搜索不便:在终端工作时需要频繁切换浏览器窗口 * 数据获取限制:百度官方API需要复杂的申请流程和使用限制 baidusearch项目正是为解决这些问题而生!这是一个开源的Python百度搜索接口封装,无需任何API密钥,安装即可使用,完美支持Python 2和3。 💡 核心解决方案:零配置搜索引擎 免密钥设计 与大多数需要复杂申请的搜索API不同,baidusearch采用网页爬取技术,直接模拟浏览器

By Ne0inhk

紧急避坑指南:Python中生成真随机数的3种安全方式,第2种最推荐

第一章:Python中随机数生成的基本概念 在编程中,随机数被广泛应用于模拟、游戏开发、密码学和机器学习等领域。Python 提供了内置的 `random` 模块,用于生成伪随机数。这些数字并非真正意义上的“随机”,而是通过确定性算法生成的序列,称为伪随机数。 随机数生成器的工作原理 Python 的 `random` 模块基于梅森旋转算法(Mersenne Twister)生成随机数,该算法具有极长的周期(2¹⁹⁹³⁷−1),适用于大多数非加密场景。每次调用随机函数时,系统会根据当前种子值计算下一个状态,并返回对应的随机结果。 常用随机数生成方法 以下是几种常见的随机操作及其用途: * random.random():生成一个 [0.0, 1.0) 区间内的浮点数 * random.randint(a, b):返回一个 a 到 b 之间的整数(包含

By Ne0inhk
【Java 学习】详细讲解---包和导包、Scanner类、输入源

【Java 学习】详细讲解---包和导包、Scanner类、输入源

1. 包 1.1 什么是包? 举个例子,你和你的同学有不同的家庭,你们都有自己的爸爸妈妈,都有自己的家。在自己的家中你们可以按照自己爱好摆放东西,都互不干扰。但是,假如你们的家都在一起,你们就不能按照自己的喜好摆放东西了,你们之间会相互影响。 为了让每个程序之间直接有相对好的隔离,就设置了包。包其实就是一个串文件夹,在不同的文件夹中可以设置相同的名字的文件。 1.2 包的主要作用 包(Package)是一种将类和接口组织在一起的机制。 1. 命名空间管理 包提供了一种命名空间管理机制,避免类名冲突。通过将类放在不同的包中,可以确保类名的唯一性。例如,多个项目中可能会有多个名为 Logger 的类,但通过将它们放在不同的包中,可以避免命名冲突。 2. 访问控制 包还提供了访问控制机制。通过使用不同的访问修饰符(如 public、protected、默认(即不写访问修饰符)和 private),可以控制类、方法和变量的可见性。

By Ne0inhk
python+uniapp微信小程序的博物馆文创产品推荐商城销售系统

python+uniapp微信小程序的博物馆文创产品推荐商城销售系统

目录 * 技术架构设计 * 数据库模型设计 * 推荐算法实现 * 微信小程序集成 * 性能优化策略 * 安全防护措施 * 开发技术 * 源码文档获取/同行可拿货,招校园代理 :文章底部获取博主联系方式! 技术架构设计 Python后端采用Django或Flask框架,提供RESTful API接口。数据库使用MySQL或PostgreSQL存储用户信息、商品数据、订单记录。Redis缓存热门商品和用户会话信息。 Uniapp前端使用Vue.js构建跨平台应用,适配微信小程序。通过uni.request与后端API通信,实现数据交互。采用Vuex管理全局状态,如用户登录状态、购物车信息。 数据库模型设计 用户表包含字段:用户ID、用户名、密码(加密存储)、手机号、收货地址、收藏商品列表。商品表包含字段:商品ID、名称、分类、价格、库存、详情描述、图片链接、销量、评分。 订单表记录用户购买行为,字段包括:

By Ne0inhk