本文由 简悦 SimpRead 转码, 原文地址 mp.weixin.qq.com
面试官也在看的前端面试资料
给定一个二叉树,检查它是否是镜像对称的。
例如,二叉树 [1,2,2,3,4,4,3] 是对称的。
1<br style="visibility: visible;"> / \<br style="visibility: visible;"> 2 2<br style="visibility: visible;"> / \ / \<br style="visibility: visible;">3 4 4 3<br style="visibility: visible;">但是下面这个 [1,2,2,null,3,null,3] 则不是镜像对称的:
1<br style="visibility: visible;"> / \<br style="visibility: visible;"> 2 2<br style="visibility: visible;"> \ \<br style="visibility: visible;"> 3 3<br style="visibility: visible;">进阶:
你可以运用递归和迭代两种方法解决这个问题吗?
解答:
一棵二叉树对称,则需要满足:根的左右子树是镜像对称的
也就是说,每棵树的左子树都和另外一颗树的右子树镜像对称,左子树的根节点值与右子树的根节点值相等
所以,我们需要比较:
左右子树的根节点值是否相等
左右子树是否镜像对称
边界条件:
左右子树都为
null时,返回true左右子树有一个
null时,返回false
解法一:递归
const isSymmetric = function(root) { if(!root) return true var isEqual = function(left, right) { if(!left && !right) return true if(!left || !right) return false return left.val === right.val && isEqual(left.left, right.right) && isEqual(left.right, right.left) } return isEqual(root.left, root.right)};复杂度分析:
时间复杂度:O(n)
空间复杂度:O(n)
解法二:迭代
利用栈来记录比较的过程,实际上,递归就使用了调用栈,所以这里我们可以使用栈来模拟递归的过程
首先根的左右子树入栈
将左右子树出栈,比较两个数是否互为镜像
如果左右子树的根节点值相等,则将左子树的
left、右子树的right、左子树的right、右子树的left依次入栈继续出栈(一次出栈两个进行比较)…….
依次循环出栈入栈,直到栈为空
const isSymmetric = function(root) { if(!root) return true let stack = [root.left, root.right] while(stack.length) { let right = stack.pop() let left = stack.pop() if(left && right) { if(left.val !== right.val) return false stack.push(left.left) stack.push(right.right) stack.push(left.right) stack.push(right.left) } else if(left || right) { return false } } return true};复杂度分析:
时间复杂度:O(n)
空间复杂度:O(n)
来自:https://github.com/sisterAn/JavaScript-Algorithms
最后
欢迎关注「三分钟学前端」,回复「交流」自动加入前端三分钟进阶群,每日一道编程算法面试题(含解答),助力你成为更优秀的前端开发!
》》面试官也在看的前端面试资料《《
“在看和转发” 就是最大的支持