【Java--数据结构】二叉树oj题(上)

前言

欢迎关注个人主页:逸狼


创造不易,可以点点赞吗~

如有错误,欢迎指出~



判断是否是相同的树

oj链接

要判断树是否一样,要满足3个条件

  • 结构 和 值 一样
  • 左子树的结构和值一样
  • 右子树的结构和值一样

所以就可以总结以下思路:

  1. 一个为空,一个不为空--》一定不相同
  2. 两个都为空--》 相同
  3. 都不为空 ,但值不一样--》一定不相同
  4. 最后递归判断 左子树和右子树都要相同--》两棵树相同

其中该题的时间复杂度为O(min(m,n)),也就是取m和n中最小值(假设p的节点数为m个,q的节点数为n个)

复制代码
public boolean isSameTree(TreeNode p, TreeNode q) {
        //一个为空,一个不为空
        if(p!=null&&q==null||p==null&&q!=null){
            return false;
        }
        //此时要么两个都为空,要么都不为空
        if(p==null&&q==null){
            return true;
        }
        //都不为空
        if(p.val!=q.val){
            return false;
        }
        //此时两个都不为空,val值也一样,说明根节点相同
        //判断左右树是否相同
        return isSameTree(p.left,q.left)&&isSameTree(p.right,q.right);

另一棵树的子树

oj链接

当两颗树相同时,也属于子树

所以步骤如下

  1. 判断是不是两颗相同的树
  2. 若不是,有可能是子树的子树
  3. 也有可能是子树的子树

其中该题的时间复杂度为m*n (假设root有n个节点,subRoot有m个节点),原因是root的每一个节点都要和subRoot的节点比对

复制代码
    public boolean isSubtree(TreeNode root, TreeNode subRoot) {
        //因为root要递归,递归到后面root可能为空
        if(root==null){
            return false;
        }
        //两颗树相同时,成立
        if(isSameTree(root,subRoot)){
            return true;
        }
        //判断root的左子树和subRoot
        if(isSubtree(root.left,subRoot)){
            return true;
        }
        //判断root的右子树和subRoot
        if(isSubtree(root.right,subRoot)){
            return true;
        }
        return false;

    }

        public boolean isSameTree(TreeNode p, TreeNode q) {
        //一个为空,一个不为空
        if(p!=null&&q==null||p==null&&q!=null){
            return false;
        }
        //此时要么两个都为空,要么都不为空
        if(p==null&&q==null){
            return true;
        }
        //都不为空
        if(p.val!=q.val){
            return false;
        }
        //此时两个都不为空,val值也一样,说明根节点相同
        //判断左右树是否相同
        return isSameTree(p.left,q.left)&&isSameTree(p.right,q.right);

    }

翻转二叉树

oj链接

让root的左节点和右节点交换,再递归遍历root.left和root.right使左子树和右子树都翻转。

代码优化:若只有一个根节点(左右子树都为空),直接返回;减少了递归和交换的次数

复制代码
    public TreeNode invertTree(TreeNode root) {
        if(root==null){
            return null;
        }
        //代码优化部分******减少一些递归和交换的次数
        if(root.left==null&&root.right==null){
            return root;
        }
        //          ******
        TreeNode ret=root.left;
        root.left=root.right;
        root.right=ret;
        invertTree(root.left);
        invertTree(root.right);
        return root;
    }

判断一颗二叉树是否是平衡二叉树

平衡二叉树 是指该树所有 节点的左右子树的深度相差不超过 1

oj链接

判断步骤:

当前root的 左子树 和 右子树的高度差<=1

同时满足root的左 右子树平衡

其中该题的时间复杂度为O(n^2)

复制代码
    public boolean isBalanced(TreeNode root) {
        if(root==null) return true;
        int leftH=maxDepth(root.left);
        int rightH=maxDepth(root.right);

        return Math.abs(leftH-rightH)<=1
        &&isBalanced(root.left)
        &&isBalanced(root.right);
    }
    public int maxDepth(TreeNode root){
        if(root==null){
            return 0;
        }
        int leftH=maxDepth(root.left);
        int rightH=maxDepth(root.right);
        return leftH>rightH?leftH+1:rightH+1;
    }

代码优化,使得时间复杂度变为O(n)

复制代码
    public boolean isBalanced(TreeNode root) {
        if(root==null) return true;
        return maxDepth(root)>=1;
    }
    public int maxDepth(TreeNode root){
        if(root==null){
            return 0;
        }
        int leftH=maxDepth(root.left);
        if(leftH<0){
            return -1;
        }
        int rightH=maxDepth(root.right);
        if(rightH<0){
            return -1;
        }
        if(Math.abs(leftH-rightH)<=1){
            return leftH>rightH?leftH+1:rightH+1;
        }else{
            return -1;
        }

    }

第三种写法

复制代码
    public boolean isBalanced(TreeNode root) {
        if(root==null) return true;
        return maxDepth(root)>=1;
    }
    public int maxDepth(TreeNode root){
        if(root==null){
            return 0;
        }
        int leftH=maxDepth(root.left);
        // if(leftH<0){
        //     return -1; 
        // }
        int rightH=maxDepth(root.right);
        // if(rightH<0){
        //     return -1;
        // }
        if(leftH>=0&&rightH>=0
        &&Math.abs(leftH-rightH)<=1){
            return Math.max(leftH,rightH)+ 1;
        }else{
            return -1;
        }
    }
相关推荐
橘子编程2 分钟前
MindOS:你的AI第二大脑知识库
java·开发语言·人工智能·计算机网络·ai
XWalnut5 分钟前
LeetCode刷题 day9
java·算法·leetcode
bIo7lyA8v5 分钟前
算法稳定性分析中的随机扰动建模的技术9
算法
忧郁的Mr.Li6 分钟前
JAVA工具类---PDF电子签章工具类
java·pdf
aini_lovee8 分钟前
C# 快速搜索磁盘文件解决方案
开发语言·c#
小陈工8 分钟前
2026年4月8日技术资讯洞察:边缘AI推理框架竞争白热化,Python后端开发者的机遇与挑战
开发语言·数据库·人工智能·python·微服务·回归
谢白羽13 分钟前
vllm抢占机制详解
算法·vllm
Hello--_--World14 分钟前
Vue2的 双端 diff算法 与 Vue3 的 快速diff 算法
前端·vue.js·算法
零二年的冬14 分钟前
epoll详解
java·linux·开发语言·c++·链表
凭君语未可20 分钟前
Java 中的接口是什么
java·开发语言