【力扣】二叉树的最近公共祖先

简介: 【力扣】二叉树的最近公共祖先

解题思路


题目链接


首先,我们要明白什么是最近公共祖先(LCA)。在二叉树中,如果一个节点r是p和q的祖先,那么p和q都在r的子树中。如果r是p和q的最近公共祖先,那么r的深度要比其他祖先节点大,也就是说,r离p和q最近。例如,下图中的二叉树,实例1节点4和7的LCA是2,示例2节点5和4的LCA是5(注意本身也可以是祖先节点)。


4667568c87014eb186b35594e263b747.png



a76a4ab282c54b679209220398010791.png


其次,我们要知道如何找到从根节点到目标节点的路径。这里我们使用了一个递归的方法,也就是说,我们从根节点开始,依次访问左右子节点,并判断当前节点是否等于目标节点或者其左右子树中是否包含目标节点。如果是,就将当前节点压入一个栈中,并返回true;如果不是,就将当前节点弹出,并返回false。这样,当递归结束时,我们就得到了一条从根节点到目标节点的路径。


最后,我们要知道如何利用两条路径来找到LCA。这里我们使用了一个简单的思路,就是比较两个栈顶元素是否相同。如果相同,就说明找到了LCA;如果不相同,就说明还没有找到LCA,需要继续弹出栈顶元素。但是,在比较之前,我们需要保证两个栈的大小相同,因为p和q可能在不同的深度。所以我们需要将较长路径上多余的节点弹出。例如,下图中的二叉树,从根节点到节点7和0的路径分别为[3,5,2,7]和[3,4,0]。我们首先将两个栈调整为相同大小,然后比较两个栈顶元素是否相同。发现不相同,就弹出两个栈顶元素,并继续比较下一个栈顶元素。发现相同,就说明找到了LCA,即3。


ae4f1028e76840aab300d4bf448bb178.png



代码示例


/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */
class Solution {
public:
    // 定义一个函数,用于从根节点到目标节点的路径,并将路径上的节点压入栈中
    bool getPath(TreeNode* root ,TreeNode* x,stack<TreeNode*>& path)
    {
        if(root== nullptr) // 如果根节点为空,返回false
            return false;
        path.push(root); // 将根节点压入栈中
        if(root == x) // 如果根节点就是目标节点,返回true
            return true;
        if(getPath(root->left,x,path)) // 如果左子树中存在目标节点,返回true
            return true;
        if(getPath(root->right,x,path)) // 如果右子树中存在目标节点,返回true
            return true;
        path.pop(); // 如果左右子树都不存在目标节点,将根节点弹出栈,返回false
        return false;
    }
    // 定义一个函数,用于求解二叉树的最近公共祖先
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        stack<TreeNode* > pPath,qPath; // 定义两个栈,分别存储从根节点到p和q的路径
        getPath(root,p,pPath); // 调用getPath函数,求出从根节点到p的路径
        getPath(root,q,qPath); // 调用getPath函数,求出从根节点到q的路径
        while(pPath.size() != qPath.size()) // 由于p和q可能在不同的深度,所以需要将两个栈的大小调整为相同
        {
            if(pPath.size() > qPath.size()) // 如果p的路径较长,弹出多余的节点
                pPath.pop();
            else // 如果q的路径较长,弹出多余的节点
                qPath.pop();
        }
        while(pPath.top() != qPath.top()) // 当两个栈顶元素不相同时,说明还没有找到最近公共祖先,继续弹出节点
        {
            pPath.pop();
            qPath.pop();
        }
        return pPath.top(); // 当两个栈顶元素相同时,说明找到了最近公共祖先,返回该节点
    }
};


DFS:深度优先搜索算法。这是一种用于遍历或搜索树或图数据结构的算法,它从根节点开始(在图的情况下选择任意一个节点作为根节点),沿着每个分支尽可能深入地探索,直到遇到死胡同,然后回溯。这种算法通常需要额外的内存,通常是一个栈,来记录沿着指定分支发现的节点,以便在回溯时使用。

相关文章
|
6月前
|
Go 开发者 索引
【LeetCode 热题100】路径与祖先:二叉树中的深度追踪技巧(力扣33 / 81/ 153/154)(Go语言版)
本文深入探讨了LeetCode中四道关于「搜索旋转排序数组」的经典题目,涵盖了无重复和有重复元素的情况。通过二分查找的变形应用,文章详细解析了每道题的解题思路和Go语言实现代码。关键点包括判断有序区间、处理重复元素以及如何缩小搜索范围。文章还总结了各题的异同,并推荐了类似题目,帮助读者全面掌握二分查找在旋转数组中的应用。无论是初学者还是有经验的开发者,都能从中获得实用的解题技巧和代码实现方法。
306 14
|
7月前
|
算法 Go
【LeetCode 热题100】深入理解二叉树结构变化与路径特性(力扣104 / 226 / 114 / 543)(Go语言版)
本博客深入探讨二叉树的深度计算、结构变换与路径分析,涵盖四道经典题目:104(最大深度)、226(翻转二叉树)、114(展开为链表)和543(二叉树直径)。通过递归与遍历策略(前序、后序等),解析每题的核心思路与实现方法。结合代码示例(Go语言),帮助读者掌握二叉树相关算法的精髓。下一讲将聚焦二叉树构造问题,欢迎持续关注!
192 10
|
7月前
|
存储 算法 数据可视化
【二叉树遍历入门:从中序遍历到层序与右视图】【LeetCode 热题100】94:二叉树的中序遍历、102:二叉树的层序遍历、199:二叉树的右视图(详细解析)(Go语言版)
本文详细解析了二叉树的三种经典遍历方式:中序遍历(94题)、层序遍历(102题)和右视图(199题)。通过递归与迭代实现中序遍历,深入理解深度优先搜索(DFS);借助队列完成层序遍历和右视图,掌握广度优先搜索(BFS)。文章对比DFS与BFS的思维方式,总结不同遍历的应用场景,为后续构造树结构奠定基础。
374 10
|
7月前
|
Go
【LeetCode 热题100】路径与祖先:二叉树中的深度追踪技巧(力扣437 / 236 )(Go语言版)
本文深入探讨二叉树中路径与祖先问题,涵盖两道经典题目:LeetCode 437(路径总和 III)和236(最近公共祖先)。对于路径总和 III,文章分析了双递归暴力解法与前缀和优化方法,后者通过哈希表记录路径和,将时间复杂度从O(n²)降至O(n)。在最近公共祖先问题中,采用后序遍历递归查找,利用“自底向上”的思路确定最近公共祖先节点。文中详细解析代码实现与核心要点,帮助读者掌握深度追踪技巧,理解树结构中路径与节点关系的本质。这类问题在面试中高频出现,掌握其解法意义重大。
183 4
|
7月前
|
Go 索引 Perl
【LeetCode 热题100】【二叉树构造题精讲:前序 + 中序建树 & 有序数组构造 BST】(详细解析)(Go语言版)
本文详细解析了二叉树构造的两类经典问题:通过前序与中序遍历重建二叉树(LeetCode 105),以及将有序数组转化为平衡二叉搜索树(BST,LeetCode 108)。文章从核心思路、递归解法到实现细节逐一拆解,强调通过索引控制子树范围以优化性能,并对比两题的不同构造逻辑。最后总结通用构造套路,提供进阶思考方向,帮助彻底掌握二叉树构造类题目。
419 9
【LeetCode 43】236.二叉树的最近公共祖先
【LeetCode 43】236.二叉树的最近公共祖先
129 0
【LeetCode 38】617.合并二叉树
【LeetCode 38】617.合并二叉树
94 0
【LeetCode 37】106.从中序与后序遍历构造二叉树
【LeetCode 37】106.从中序与后序遍历构造二叉树
114 0
【LeetCode 34】257.二叉树的所有路径
【LeetCode 34】257.二叉树的所有路径
107 0
【LeetCode 32】111.二叉树的最小深度
【LeetCode 32】111.二叉树的最小深度
108 0

热门文章

最新文章