代码随想录刷题笔记 DAY15 | 翻转二叉树 No.226 | 对称二叉树 No.101

Day 15

01. 翻转二叉树(No. 226)

题目链接

代码随想录题解

1.1 题目

给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。

示例 1:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

输入:root = [4,2,7,1,3,6,9]
输出:[4,7,2,9,6,3,1]

示例 2:

输入:root = [2,1,3]
输出:[2,3,1]

示例 3:

输入:root = []
输出:[]

提示:

  • 树中节点数目范围在 [0, 100]
  • -100 <= Node.val <= 100
1.2 笔记

一道考察递归遍历的题目,我们要先理清这道题目的要求

题目要求我们的是交换二叉树的节点,注意不是交换值,所以我们要在递归遍历的时候去交换节点,递归遍历分为前序、中序和后序,我们只需要在遍历到节点的期间去交换节点即可。

下面我们以前序为例子:

和上面提到的相同,前序遍历就是在进入节点的时候立刻进行的操作,我们在进入节点前交换左右子树,和上一节博客讲的一样,对所有的节点进行相同的处理,即可完成交换

后序遍历也是同理。

但是中序遍历是不行的,中序遍历是在左子树返回的时候进行交换,我们看看会有什么后果:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

比如我们这里把左子树处理完了

这时候我们中序回到根节点,将 2 和 7 一交换再去递归右边,这不是就完全恢复了吗?

就相当于我们只交换了根节点下面的两个节点,其他是没变化的

写到这里就可以给出代码了

1.3 代码
class Solution {
    public TreeNode invertTree(TreeNode root) {
        traverse(root);
        return root;
    }
    public void traverse(TreeNode root) {
        // 递归出口
        if (root == null) {
            return;
        }
        // 交换节点
        TreeNode temp = root.left;
        root.left = root.right;
        root.right = temp;
        
        traverse(root.left);
        traverse(root.right);
    }
}

02. 对称二叉树(No. 101)

题目链接

代码随想录题解

2.1 题目

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

示例 1:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

输入:root = [1,2,2,3,4,4,3]
输出:true

示例 2:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

输入:root = [1,2,2,null,3,null,3]
输出:false

提示:

  • 树中节点数目在范围 [1, 1000]
  • -100 <= Node.val <= 100
2.2 笔记

这道题我们要理解后序遍历的特殊性,后序遍历是在遍历完子节点后返回时做的操作,这就意味着我们可以 收集到子节点的信息,所以后序遍历经常用在需要子节点信息的情况。

这道题我们要比较内侧和外侧的信息,我们如何遍历这个树的外侧呢?

回顾一下我们写的二叉树遍历的代码:

public void reverse(TreeNode node) {
	reverse(node.right);
	reverse(node.left);
}

我们对每个节点进行的操作是遍历 左子树 和 右子树,但如果我们要求的只是遍历外侧,也就是说只遍历 左节点的左节点和右节点的右节点:

public void reverse(Treenode right) {
	reverse(node.right);
}

public void reverse(Treenode left) {
	reverse(node.left);
}

这便是遍历树的外侧,抛去二叉树的外壳,这其实就是链表的递归遍历

接下来我们将这两个方向的遍历合在一起

    public boolean isSymmetric(TreeNode root) {
        return compare(root.left, root.right);
    }
    public boolean compare(TreeNode left, TreeNode right) {
        if (left == null && right != null) {
            return false;
        }
        if (right == null && left != null) {
            return false;
        }
        if (right == null && left == null) {
            return true;
        }
        if (right.val != left.val) {
            return false;
        }
        boolean compareOutside = compare(left.left, right.right);
        return compareOutside;
}

上述的代码就是将根节点的左右节点分别传入,对左节点只进行向左的递归,对右节点只进行向右的递归,注意我们返回结果的位置正是说的后序的位置,因为只有这个位置才是验证完成 子节点 的对称性并且返回的的位置。

这个位置可以理解为我们 收集信息 的位置,如果对理解递归有困难的话,我们可以想象有一个外置的 boolean flg 全局变量,当我们一旦发现 左右不对称的时候,也就是我们的 compare(left.left, right.right); 返回值为 false 的时候,就将这个 flg 设置为 false,实际上这和递归实现的效果是相同的。

那对于内侧的遍历也很好写出来了:

class Solution {
    public boolean isSymmetric(TreeNode root) {
        return compare(root.left, root.right);
    }
    public boolean compare(TreeNode left, TreeNode right) {
        if (left == null && right != null) {
            return false;
        }
        if (right == null && left != null) {
            return false;
        }
        if (right == null && left == null) {
            return true;
        }
        if (right.val != left.val) {
            return false;
        }
        boolean compareInside = compare(left.right, right.left);
        return compareInside;
    }
}

内侧就是首先传入根节点的 左子树,对左子树向右递归遍历,以及对右子树向左递归遍历

将这两段代码整合起来,就变成了这道题的解答

2.3 代码
/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode() {}
 *     TreeNode(int val) { this.val = val; }
 *     TreeNode(int val, TreeNode left, TreeNode right) {
 *         this.val = val;
 *         this.left = left;
 *         this.right = right;
 *     }
 * }
 */
class Solution {
    public boolean isSymmetric(TreeNode root) {
        return compare(root.left, root.right);
    }
    public boolean compare(TreeNode left, TreeNode right) {
        if (left == null && right != null) {
            return false;
        }
        if (right == null && left != null) {
            return false;
        }
        if (right == null && left == null) {
            return true;
        }
        if (right.val != left.val) {
            return false;
        }
        boolean compareOutside = compare(left.left, right.right);
        boolean compareInside = compare(left.right, right.left);
        return compareInside && compareOutside;

    }
}
  • 13
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

*Soo_Young*

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值