分析&回答
利用后序遍历实现;
对于当前节点cur,如果节点为null或者等于n1或n2中的一个,则直接返回cur;
先处理左右子树,左子树返回left,右子树返回right;判断left和right;
- left和right均为null,说明以cur为根的树上没发现n1和n2;
- left和right均不为null,说明在左子树上发现了n1或n2,在右子树上也发现了n1或n2,cur为n1和n2的首次相遇节点,则直接返回cur;
- left和right中一个null,另一个不为null,说明不为空的节点是n1和n2的其中一个,或者是n1和n2的最近公共祖先;则返回不为空的节点;
图解
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5xOY0jYr-1651804399569)(https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/257ce78acd414824abad8da45b582238~tplv-k3u1fbpfcp-watermark.image?)]
编码
public class LowestAncestor {
/*求解n1和n1的最近公共祖先
*
* 利用后序遍历求解
*/
public Node getLowestAncestor(Node head,Node n1,Node n2){
if(head==null || head==n1 || head==n2){
return head;
}
//先遍历左右子树,左右子树的返回为left和right;然后判断left和right请情况
Node left=getLowestAncestor(head.left,n1,n2);
Node right=getLowestAncestor(head.right,n1,n2);
/*左和右都不为null---说明在左子树中发现过n1或n2,
* 在右子树上也发现过n1或n2,并且n1和n2在当前节点首次相遇
*/
if(left!=null && right!=null){
return head;
}
/*左和右中一个不为null,另一个为null,
* 说明不为null的节点是n1或n2中的一个,或者是n1和n2的最近祖先;
* 直接返回;
*/
if(left!=null){
return left;
}
if(right!=null){
return right;
}
//左和右均为null,没有发现n1和n2;
return null;
}
}
反思&扩展
喵呜面试助手: 一站式解决面试问题,你可以搜索微信小程序 [喵呜面试助手] 或关注 [喵呜刷题] -> 面试助手 免费刷题。如有好的面试知识或技巧期待您的共享!