【每天学习一点算法 2025/12/18】对称二叉树

每天学习一点算法 2025/12/18

题目:对称二叉树

给你一个二叉树的根节点 root , 检查它是否轴对称。

  1. 老规矩先来递归, 检查二叉树是否轴对称,我们是不是得比较根节点下面得两个子树是否是轴对称,那么我们只需要递归比较两个子树的值即可,那要怎么传参呢?首先肯定是要将根节点的两个子节点作为初始传参传进去的,那么我们需要一个辅助函数,然后每次递归对称传递子结点比较。

    function isSymmetric(root: TreeNode | null): boolean {
      if (!root) return true; // 空树默认为true
      // 定义辅助函数 递归比较两个子树
      function validate(left: TreeNode | null, right: TreeNode | null): boolean {
        // 两个子节点同时为空才有轴对称的可能
        if (!left && !right) return true;
        if (!left || !right) return false;
        // 比较左右子节点的值,并传递对称的子节点递归
        return left.val === right.val && validate(left.left, right.right) && validate(left.right, right.left);
      }
    	// 调用辅助函数
      return validate(root.left, root.right);
    };
    
  2. 还有我们是不是可以用层序遍历的方法取得每一层的节点然后我们校验他们是否是对称就行了。核心思想其实跟递归是一样的,找到对称节点作比较。

    如果我们拓展下一层按照左左↔右右、左右↔右左的顺序入队列,那么每两个节点都是对称位置上的节点了,比如说我们有一个四层的结构对称的树,像下面这个样。

                1
            /       \
         2            3
        /   \       /   \
       4     5     6     7
      / \   / \   / \   / \
     8   9 10 11 12 13 14 15
    
    • 我们初始两个子节点入队列: [2, 3]
    • 然后 2, 3 出队列,子节点按照左左↔右右、左右↔右左的顺序入队列:[4, 7, 5, 6]
    • 然后 4, 7 出队列,子节点按照左左↔右右、左右↔右左的顺序入队列:[5, 6, 8, 15, 9, 14]
    • 然后 5, 6 出队列,子节点按照左左↔右右、左右↔右左的顺序入队列:[8, 15, 9, 14, 10, 13, 11, 12]
    • 然后剩下的子节点两两出队列。

    可以看到整个循环的过程子节点的遍历是从上到下,每层从两端向中间收拢的,我们按照这种方式遍历比较两两出队列的节点值就可判断出树是否对称。

    function isSymmetric(root: TreeNode | null): boolean {
      // 空树默认为对称
      if (!root) return true;
      // 初始化队列,用于层序遍历对称位置的节点
      const queue: Array<TreeNode | null > = [];
    
      // 根节点的左右子节点都为空,是对称树
      if (!root.left && !root.right) return true;
      // 根节点的左右子节点只有一个为空,必然不对称
      if (!root.left || !root.right) return false;
    
      // 将根节点的左右子节点(对称的起始节点)入队
      queue.push(root.left);
      queue.push(root.right);
    
      // 层序遍历队列中的对称节点对
      while (queue.length) {
        // 每次取出队列前两个节点(一对对称节点)
        const left = queue.shift()!;
        const right = queue.shift()!;
    
        // 两个节点都为空,说明当前对称位置无节点,继续检查下一对
        if (!left && !right) continue;
    
        // 不对称的情况:
        // 1. 其中一个节点为空,另一个非空
        // 2. 两个节点值不相等
        if ((!left || !right) || (left.val !== right.val)) return false;
    
        // 按对称规则将下一层节点入队:
        // 左节点的左子节点对应右节点的右子节点
        queue.push(left.left);
        queue.push(right.right);
        // 左节点的右子节点对应右节点的左子节点
        queue.push(left.right);
        queue.push(right.left);
      }
    
      // 所有对称节点对都验证通过,说明是对称二叉树
      return true;
    }
    

题目来源:力扣(LeetCode)

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值