- 03. 数组中重复的数字
- 04. 二维数组中的查找
- 05.替换空格
- 06.从尾到头打印链表
- 07.重建二叉树
- 09.用两个栈实现队列
- 10-1.斐波那契数列
- 10-2.青蛙跳台阶问题
- 剑指 offer 11.旋转数组的最小数字
- 剑指offer 12.矩阵中的路径
- 剑指offer 13.机器人的运动范围
- 剑指offer 14-1.剪绳子
- 剑指offer 14-2.剪绳子2
- 剑指offer 15.二进制中1的个数
- 剑指offer 16.数值的整数次方
- 剑指offer 17.打印从1到最大的n位数
- 剑指offer 18.删除链表的节点
- 剑指offer 19.正则表达式匹配
- 剑指offer 20.表示数值的字符串
- 剑指offer 21.调整数组顺序使奇数位于偶数前面
- 剑指offer 22.链表中倒数第k个节点
- 剑指offer 24.反转链表
- 剑指offer 25.合并两个排序的链表
- 剑指offer 26.树的子结构
- 剑指offer 27.二叉树的镜像
- 剑指offer 28.对称的二叉树
- 剑指offer 29.顺时针矩阵
- 剑指offer 30.包含min函数的栈
- 剑指offer 31.栈的压入、弹出序列
- 剑指offer 32-1.从上到下打印二叉树
- 剑指offer 32-2.从上到下打印二叉树
- 剑指offer 32-3.从上到下打印二叉树
- 剑指offer 33.二叉搜索树的后序遍历序列
- 剑指offer 34.二叉树中和为某一值的路径
- 剑指offer 35.复杂链表的复制
- 剑指offer 36.二叉搜索树与双向链表
- 剑指offer 37.序列化二叉树
- 剑指offer 38.字符串的排列
- 剑指 offer 39. 数组中出现次数超过一半的数字
- 剑指offer 40. 最小的k个数
- 剑指offer 41.数据流中的中位数
- 剑指offer 42. 连续子数组的最大和
- 剑指offer 43. 1~n整数中1出现的次数
- 剑指offer 44. 数字序列中某一位的数字
- 剑指offer 45. 把数组排成最小的数
- 剑指offer 46.把数字翻译成字符串
- 剑指offer 47.礼物的最大值
- 剑指offer 48.最长不含重复字符的字符子串
- 剑指offer 49.丑数
- 剑指offer 50.第一个只出现一次的字符
- 剑指offer 51. 数组中的逆序对
- 剑指offer 52.两个链表的第一个公共节点
- 剑指offer 53-1. 在排序数组中查找数字
- 剑指offer 53-2. 0~n-1中缺失的数字
- 剑指offer 54. 二叉搜索树的第k大节点
- 剑指offer 55-1. 二叉树的深度
- 剑指offer 55-2. 平衡二叉树
- 剑指offer 56-1. 数组中数字出现的次数
- 剑指offer 56-2. 数字中出现的次数
- 剑指offer 57-1. 和为s的两个数字
- 剑指offer 57-2. 和为s的连续正数序列
- 剑指offer 58-1. 翻转单词顺序
- 剑指offer 58-2.左旋转字符串
- 剑指offer 59-1. 滑动窗口的最大值
- 剑指offer 59-2. 队列的最大值
- 剑指offer 60. n个骰子的点数
- 剑指offer 61. 扑克牌中的顺子
- 剑指offer 62. 圆圈中最后剩下的数字
- 剑指offer 63. 股票的最大利润
- 剑指offer 64. 求1+2+…+n
- 剑指offer 65. 不用加减乘除做加法
- 剑指offer 66. 构建乘积数组
- 剑指offer 67. 把字符串转成整数
- 剑指offer 68-1. 二叉搜索树的最近公共祖先
- 剑指offer 68-2. 二叉树的最近公共祖先
剑指offer 68-2. 二叉树的最近公共祖先
题目描述
给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。
百度百科中最近公共祖先的定义为:“对于有根树 T 的两个结点 p、q,最近公共祖先表示为一个结点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大( 一个节点也可以是它自己的祖先 )。”
例如,给定如下二叉树: root = [3,5,1,6,2,0,8,null,null,7,4]
示例 1:
输入: root = [3,5,1,6,2,0,8,null,null,7,4], p = 5, q = 1
输出: 3
解释: 节点 5 和节点 1 的最近公共祖先是节点 3。
示例 2:
输入: root = [3,5,1,6,2,0,8,null,null,7,4], p = 5, q = 4
输出: 5
解释: 节点 5 和节点 4 的最近公共祖先是节点 5。因为根据定义最近公共祖先节点可以为节点本身。
说明:
- 所有节点的值都是唯一的。
- p、q 为不同节点且均存在于给定的二叉树中。
题解
(递归) O(n)
求公共祖先需要自底向上递归遍历二叉树,那么可以按照后序顺序遍历。
题目已知 p 和 q 一定在二叉树中,所以一定存在非空答案。
递归三部曲:
- 递归函数的含义:直接使用题目给出的函数,返回以 root 为根节点二叉树中节点 p 和 q 的最近公共祖先,返回值就是最近公共祖先。返回值总共有四种情况:
- 如果以 root 为根的子树中包含 p 和 q,则返回它们的最近公共祖先即 root
- 如果只包含 p,返回 p
- 如果只包含 q,返回 q
- 如果都不包含,则返回 NULL
- 递归出口条件:如果当前节点 root 为空或等于 p 或等于 q,直接返回当前节点即可。
- 如果
root == NULL
说明没找到 p 或 q,返回 NULL 即可 - 如果
root == p
说明当前树包含 p,p 是它自己的最近公共祖先,返回 p - 如果
root == q
说明当前树包含 q,q 是它自己的最近公共祖先,返回 q
- 如果
- 单层递归逻辑:递归查找左子树中 p 和 q 的最近公共祖先 left,递归查找右子树中 p 和 q 的最近公共祖先 right,所以对于 left 和 right 有四种情况:
- 如果 left 为空,最近公共祖先一定在右子树中,返回 right 即可。
- 如果 right 为空,最近公共祖先一定在左子树中,返回 left 即可。
- 如果 left 和 right 都不为空,则返回当前节点
- 如果 left 和 right 都为空,则返回空,这种情况可以在第一种情况中得到处理。
时间复杂度
二叉树中的每个节点只需被遍历一次,所以时间复杂度为 O(n)。
空间复杂度
最坏情况下二叉树呈链状,需要 O(n) 的空间。
C++ 代码
/**
* 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:
TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
if (p == root || q == root || !root) return root;
auto left = lowestCommonAncestor(root->left, p, q);
auto right = lowestCommonAncestor(root->right, p, q);
if (!left) return right;
if (!right) return left;
return root;
}
};
Java 代码
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode(int x) { val = x; }
* }
*/
class Solution {
public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
if (root == null || p == root || q == root) {
return root;
}
TreeNode left = lowestCommonAncestor(root.left, p, q);
TreeNode right = lowestCommonAncestor(root.right, p, q);
if (left == null) {
return right;
}
if (right == null) {
return left;
}
return root;
}
}
Python 代码
# Definition for a binary tree node.
# class TreeNode:
# def __init__(self, x):
# self.val = x
# self.left = None
# self.right = None
class Solution:
def lowestCommonAncestor(self, root: TreeNode, p: TreeNode, q: TreeNode) -> TreeNode:
if not root or p == root or q == root:
return root
left = self.lowestCommonAncestor(root.left, p, q)
right = self.lowestCommonAncestor(root.right, p, q)
if not left:
return right
if not right:
return left
return root
本文由读者提供,Github地址:https://github.com/tonngw
点击面试手册,获取本站面试手册PDF完整版