颠沛流离学二叉树(完结撒花篇)

本篇会加入个人的所谓鱼式疯言

❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言

而是理解过并总结出来通俗易懂的大白话,

小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的.

🤭🤭🤭可能说的不是那么严谨.但小编初心是能让更多人能接受我们这个概念 !!!

在这里插入图片描述

前言

在上两篇 二叉树系列 的文章中,我们学习了二叉树的 概念特性遍历方式 , 而在本篇文章中,将展开对 二叉树其他基本核心操作 的详解 ❤️ ❤️ ❤️ ❤️

提前透露一下哦, 本篇文章的基本操作,小编都会带着大家用 递归非递归 的代码带着大家走一遍哦 💖 💖 💖 💖

目录

  1. 获取树中节点的个数

  2. 获取叶子节点的个数

  3. 获取第 K 层的节点的个数

  4. 获取二叉树的高度(最大深度)

  5. 查找 节点的值等于的 value 的节点

  6. 判断一棵树是否是完全二叉树

一. 获取树中节点的个数

1. 操作分析

由于我们的二叉树是分为 两边的,分别是 左子树右子树 , 要求树中的节点,我们就需要知道 左子树的节点右子树的节点 ,最后加上自身的 1个根节点

在这里插入图片描述

2. 代码实现

<1>. 递归实现

// 递归计算节点个数
public int Size(TreeNode root) {
    if (root==null) {
        return 0;
    }

    return Size(root.left)+1+Size(root.right);
}

递归的代码实现,就是咱们 动画上的执行过程 了,小编在这里就不赘述了 💞 💞 💞 💞

动画展示

请添加图片描述

在这里插入图片描述

<2>. 非递归实现



// 非递归遍历节点个数
   public int nodeSize;

    public void setNodeSize(TreeNode root) {
        if (root==null) {
            return;
        }

        // 只要遍历过的节点
        // 就让 节点数 ++
        nodeSize++;


        // 先遍历左子树
        setNodeSize(root.left);

        // 遍历右子树
        setNodeSize(root.right);
    }
    

在这里插入图片描述

非递归的执行过程 :

我们这里利用了 前序遍历 的思路, 当我们每走过一个节点,就让我们的 nodeSize++

当完全的走过 每一个节点 , 我们 的 nodeSize 就加多少次

当我们遍历完 整个二叉树 , 也就统计出二叉树 所有的节点

鱼式疯言

关于求树中接节点个数的操作,小编最大的体会就是 子问题递归思路

由于我们 递归 的是不断把 每一个节点重新看成是根节点 的思路

所以我们只需要求出每个根节点的个数。所累积的个数,不就是我们的 节点总数 吗 ! ! !

这就是 子问题 的思路, 小伙伴在思考这一类问题时,一定要学会把一个 大问题 拆分成我们能解决的 一个一个小问题 的思想.

二. 获取叶子节点的个数

1. 操作分析

小伙伴都知道我们 二叉树的叶子节点 的定义,就是 度为0 的节点

那么我们该怎么找 度为零 的节点呢 💞 💞 💞 💞

其实很简单,我们只需要返回 左右子树所有节点 同时 左节点右子树 也为 null

最后 左右子树的叶子节点相加 即可

在这里插入图片描述

2. 代码实现

<1>. 递归实现

// 递归计算叶子个数
public int leafSize(TreeNode root) {
    if (root==null) {
        return 0;
    }

    if (root.left==null && root.right==null) {
        return 1;
    }

    return leafSize(root.left)+leafSize(root.right);
}

在这里插入图片描述

关于 递归实现 的执行步骤,下面就让小编用 动画 的形式来描述吧 😊 😊 😊 😊

动画展示

请添加图片描述

<2>. 非递归实现


 public int leafSize;


  // 非递归求叶子节点的个数
    public void setLeafSize(TreeNode root) {
        if (root==null) {
            return;
        }

        // 如果左右节点为 null
        // 就是我们需要的节点
        if (root.right==null && root.left==null) {
            leafSize++;
        }

        // 遍历左子树
        setLeafSize(root.left);

        // 遍历右子树
        setLeafSize(root.right);
    }

  

在这里插入图片描述

非递归的实现过程

非递归的根本是依赖于 我们的遍历 来实现的,当遍历到某个 左节点为 null 并且 右节点也为 null 时, 就令 leafSize ++

当遍历完整个 二叉树的所有节点,也就意味着 所有的叶子节点 已经被 leafSize 统计完了

鱼式疯言

对于 递归实现 的思路来说,我们还是用 子问题 的思路,把一棵树拆成 多颗子树

子树 又成为 新的树, 在新的树里面去寻找 左右子树都为null 的节点 ,

最终回归到 最开始的整棵树 上解决该问题 ! ! !

三. 获取第 K 层的节点的个数

1. 操作分析

要得到 第 K 层的节点 , 不免想起我们上一篇文章中所用的 层序遍历 的思想, 每一层也就相当于 二维数组中的每一行

所有要得到 每一层的节点数 ,本质上就是当我们 递归往下递的每走一行的 左子树和右子树 , k 就 减去一行 。

最终 当 K = 1 时, 就回退到 整棵树的根节点 ,就是我们要 获取到 第 K 层的节点个数

在这里插入图片描述

2. 代码实现

 // 求树某深度的节点数
    public int treeIndexlength(TreeNode root , int key) {
        // 如果节点为 null 返回 0
        if (root==null) {
            return 0;
        }

        /**
         *
         * 当 k = 1 时 就意味着到底最后一层
         * 返回当前根节点即可
         *
         *
         *
         */

        // 如果 这个位置 没有左右
        if (key==1) {
            return 1;
        }


        /**
         * 先统计左边的节点
         *
         * 再统计右边的节点
         *
         * 最后都要减少 key 次
         * 代表统计了一层
         *
         * 也就是说每统计一层 ,就减少一层
         *
         */

        //
        int left = treeIndexlength(root.left,key-1);

        // 再统计右边的节点
        int right = treeIndexlength(root.right,key-1);



        // 一直返回该 key 位置的节点
        return left+right;
    }

在这里插入图片描述

具体 递归的过程 我们通过下面动画展示:请添加图片描述

鱼式疯言

扩展一下

小伙伴有没有想到一个点:

   // 一直返回该 key 位置的节点
    return left+right;

我们这边是 直接返回 left + right 是得到我们当前 第k层 深度的节点数

但如果我们

 return left+right + 1;  

我们如果是 + 1 呢?

答案就是返回 从整棵树的根节点第k层所有节点总和

感兴趣的小伙伴可以自己 研究 哦, 小编在这里就不细讲了 💖 💖 💖 💖**(代码如下)**

// 求树某深度的到根节点所有节点数
public int treeIndexSize(TreeNode root , int key) {
    if (root==null) {
        return 0;
    }

    if (key==1) {
        return 1;
    }


    int left=treeIndexSize(root.left,key-1);


    int right=treeIndexSize(root.right,key-1);


    // key 位置开始查找节点 并 一直累加
    return left+right+1;
}

四.获取树的高度(最大深度)

1. 操作分析

要获取高度小伙伴们先得理解高度该怎么来, 我们可以设想一下,

如果一颗树的左子树高度为 3 右子树的高度为 4 , 那么 这颗树的高度是 3 还是 4

很显然 这都 不对

树的高度最终应该是 5 才对, 因为啊, 我们树的高度是 左右子树中较高的那个高度 再 + 上自身根节点的高度 1 , 所以我们得到的树的高度是 4 + 1 才对

在这里插入图片描述

2. 代码实现

// 返回树的最大深度
public int treeHeight(TreeNode root) {
    if (root==null) {
        return 0;
    }

    // 求左子树的深度
    int leftnum=treeHeight(root.left);

    // 求右子树的深度
    int rightnum=treeHeight(root.right);

    /**
     * 根据左右子树两边的高度
     *
     * 哪边高 取 哪边
     *  最后还要加上该节点的高度
     *
     */


    return leftnum>rightnum? leftnum+1: rightnum+1;
}

在这里插入图片描述

具体我们由于递归不好描述,请小伙伴们看动画分析递归的全过程哦 💫 💫 💫 💫

动画展示

请添加图片描述

鱼式疯言

当我们用 子问题 的思路 在 寻找数的高度 的关键在于我们要注意加上不仅要比较 两端左子树右子树高度, 那个自身的 根节点的 1 也要同时加上

五. 查找节点的值等于的 value 的节点

1. 操作分析

查找某个等于 val 的值, 我们就需要在 遍历的基础上 完成

所以我们只需要考虑 一点 即可,当我们在遍历过程

如果 左子树或者右子树 的节点的 val 找到了等于 val 就返回该 节点 , 如果没找到就返回 null .

在这里插入图片描述

2. 代码实现

// 查找节点
    public TreeNode find(TreeNode root, char val) {
        if(root==null) {
            return null;
        }



        // 判断新的根节点的 val 是否 等于目标值 val
        if (Character.compare(val,root.val)==0) {
            return  root;
        }


        // 在左子树中查找该节点
        TreeNode node1=  find(root.left,val);

        // 一旦找到就返回该节点
        if (node1 !=null) {
            return node1;
        }

        // 在右子树中查找该今节点
        TreeNode node2= find(root.right,val);

        // 一点找到就返回该节点
        if (node2 !=null) {
            return node2;
        }

        // 都没有找到就返回 null
       return  null;

    }

在这里插入图片描述

下面我们看看动画对于这段代码的理解吧 🤔 🤔 🤔 🤔

动画展示

请添加图片描述

鱼式疯言

细节处理:

   // 在左子树中查找该节点
    TreeNode node1=  find(root.left,val);

    // 一旦找到就返回该节点
    if (node1 !=null) {
        return node1;
    }

    // 在右子树中查找该今节点
    TreeNode node2= find(root.right,val);

    // 一点找到就返回该节点
    if (node2 !=null) {
        return node2;
    }

我想肯定有小伙伴会直接返回 node1 或者 node2 的值, 而不加判断, 这样做是不是有点问题呢 ???

是的,是有些问题,小伙伴应该知道如果我们先 递归左子树 或者 右子树 来达到遍历的效果, 一但 左子树没有找到该 val目标值 ,就会去右边的 右子树

p但我们 中途一旦返回,就会陷入只找寻 左边一个节点 ,然后就直接返回了,未达到遍历的效果 , 也就更不可能找到我们的 val 值了

六. 判断一棵树是否是完全二叉树

1. 操作分析

要考虑是否是完全二叉树,我们只需要考虑一点即可:

左子树 < 右子树高度 : 用 -1 来标记,代表 不是完全二叉树

左子树 >= 右子树高度: 就返回该高度

在这里插入图片描述

2. 代码实现

/**
     * 检查左子树和右子树的高度关系
     * 
     * 当左子树 > 右子树时 ,就不是完全二叉树
     * 否则就是完全二叉树
     * @param root 根节点
     * @return 返回布尔类型
     * 
     */

    private int CheckTreeCount(TreeNode root) {
       if (root==null) {
           return  0;
       }

       int left=CheckTreeCount(root.left);

        /**
         * 当此树不是完全二叉树的就返回 负数
         * 
         * 而这里的判断是为了连接上一个递归下来的根节点
         * 
          */
        
        
        // 当左子树不是完全二叉树时,就返回 -1 
        if (left < 0) {
           return -1;
       }
       
       int right=CheckTreeCount(root.right);
       
        // 当右子树不是完全二叉树时, 就返回 -1 
       if(right < 0) {
            return  -1;
        }

       
       // 当左子树的高度 < 右子树的高度
        // 就说明不是完全二叉树
        if (left < right) {
            return -1;
        } else {
            
            // 该节点是完全二叉树时,就返回当前高度
            return Math.max(left,right)+1;
        }

   }

在这里插入图片描述

具体实现,请小伙伴们观看下面动态理解哦 ❣️ ❣️ ❣️ ❣️

请添加图片描述

鱼式疯言

int left=CheckTreeCount(root.left);

        /**
         * 当此树不是完全二叉树的就返回 负数
         * 
         * 而这里的判断是为了连接上一个递归下来的根节点
         * 
          */
        
        
        // 当左子树不是完全二叉树时,就返回 -1 
        if (left < 0) {
           return -1;
       }
       
       int right=CheckTreeCount(root.right);
       
        // 当右子树不是完全二叉树时, 就返回 -1 
       if(right < 0) {
            return  -1;
        }

细节处理

这里有小伙伴就问了,我可以把

   if (left < 0) {
           return -1;
       }

if(right < 0) {
        return  -1;
    }

去掉吗?

答案是 不行

因为当我们一旦判断出 有一颗子树 的 左子树的高度 > 右子树的高度 时 , 就必须返回 -1 ,然后就必须回退到上一个 根节点

如果没有及时返回 -1 ,就有可能造成 max(-1,-1) 的情况,这种情况是 不可预料 的,所以我们呢需要 及时返回

总结

  1. 获取树中节点的个数: 我们通过都用了 子问题遍历 的获取节点的个数,但子问题要 注意的是需要加上自身的那个 节点数的 1

  2. 获取叶子节点的个数 : 获取叶子节点,我们主要是根据它自身的特点 (左右节点是否同时为 null ) ,成立时我们就返回 当前节点 .

  3. 获取第 K 层的节点的个数 : 获取 第K层的节点数 的思路在遍历的情况下,每走一层,k就减去1 , 当 k= 1 时就到达该层,返回 该层节点数 即可

  4. 获取二叉树的高度(最大深度):求树的高度,本质是抓住我们左子树和右子树的 较高 的那个高度,然后 加上 自身的高度1 即可

  5. 查找 节点的值等于的 value 的节点 :查找 val 是在遍历的情况下完成的,只需要在 边遍历边查找一旦找到就返回该节点即可

  6. 判断一棵树是否是完全二叉树 : 完全二叉树的判断是建立在获取二叉树高度的思路上进行变通的, 根据完全二叉树成立条件: 左子树 >= 右子树 ,这个特点来判断即可

如果觉得小编写的还不错的咱可支持 三连 下 (定有回访哦) , 不妥当的咱请评论区 指正

希望我的文章能给各位宝子们带来哪怕一点点的收获就是 小编创作 的最大 动力 💖 💖 💖

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/681101.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

mac M1下安装PySide2

在M1下装不了PySide2, 是因为PySide2没有arm架构的包 1 先在M1上装qt5 安装qt主要是为了能用里面的Desinger, uic, rcc brew install qt5 我装完的路径在/opt/homebrew/opt/qt5 其中Designer就是用来设计界面的 rcc用resource compiler, 编绎rc资源文件的, 生成对应的py文件…

使用pexpect检查SSH上的文件是否存在

使用 pexpect 模块可以在 Python 中执行命令并检查其输出。你可以使用 ssh 命令连接到远程服务器&#xff0c;并执行 ls 命令检查文件是否存在。下面我就列举几个我经常遇到的几个错误并做个详细的解决方案。 1、问题背景 用户需要编写一个 Python 脚本&#xff0c;以检查一个…

编制教师违约金一般是多少钱

老师们&#xff0c;你们在签订合同时&#xff0c;对合同中提到的违约金条款感到疑惑&#xff1f;那么&#xff0c;编制教师的违约金一般是多少呢&#xff1f;可能很多老师在签订合同时都没有一个明确的答案。 违约金的设定是为了保障双方的权益&#xff0c;当一方违反合同约定时…

有极性电容器的引脚极性怎么判别?

由于有极性电容器有正、负之分&#xff0c;在电路中又不能乱接&#xff0c;所以在使用有极性电容器前需要先判别出正、负极。有极性电容器的正、负极判别方法如图2—9&#xff5e;图2—11所示。 方法一&#xff1a;对于未使用过的新电容&#xff0c;可以根据引脚长短来判别。引…

【阿里前端面试题】聊聊前端性能优化的方案,解决过什么样的性能问题?

大家好&#xff0c;我是“寻找DX3906”。每天进步一点。日积月累&#xff0c;有朝一日定会厚积薄发&#xff01; 前言&#xff1a; 前面已经和大家分享了4篇面试题&#xff1a; 《【阿里前端面试题】浏览器的加载渲染过程》 《【阿里前端面试题】客户端和服务器交互&#xff…

VUE3 学习笔记(12):对比Vuex与Pinia状态管理的基本理解

在组件传值中&#xff0c;当嵌套关系越来越复杂的时候必然会将混乱&#xff0c;是否可以把一些值存在一个公共位置&#xff0c;无须传值直接调用呢&#xff1f;VUEX应运而生&#xff0c;但是从VUE3开始对VUEX的支持就不那么高了&#xff0c;官方推荐使用Pinia。 Vuex配置 ST1:…

NGINX之location和rewrite

一.NGINX常用的正则表达式 二.Location location作用:对访问的路径做访问控制或者代理转发 1.location 常用的匹配规则&#xff1a; 进行普通字符精确匹配&#xff0c;也就是完全匹配^~ / 表示普通字符匹配。使用前缀匹配。如果匹配成功&#xff0c;则不再匹配其它 …

场外个股期权的杠杆率有多大?

场外个股期权作为金融衍生品的一种&#xff0c;为投资者在个股交易中提供了显著的杠杆效应。这种杠杆效应&#xff0c;简单来说&#xff0c;就是投资者通过较少的资金控制更大金额的投资&#xff0c;从而放大了投资收益的可能性&#xff0c;但同时也伴随着风险的增加。 文章来…

算法基础之哈希表

大家好&#xff0c;这里是教授.F 什么是哈希表&#xff1a; 哈希表其实就是数组的pro版本。数组有下标&#xff0c;每个下标对应着一个值。哈希表也类似&#xff0c;哈希表有很多哈希值&#xff0c;然后每一个哈希值都会对应着一个值。就是这样&#xff1a;hash(key) 哈希表的…

这个高考作文满分的极客,想和你聊聊新媒体写作

计育韬 曾为上海市高考作文满分考生 微信官方 SVG AttributeName 开发者 新榜 500 强运营人 复旦大学青年智库讲师 浙江传媒学院客座导师 上海团市委新媒体顾问 上海市金山区青联副主席 文案能力&#xff0c;从来就不是一蹴而就的。今天&#xff0c;来和大家聊聊当年我的…

端午档新片速递《谈判专家》领衔,每日影视作品推荐❗❗❗多部佳作待映

每日影视作品推荐一、新片速递《谈判专家》上映时间&#xff1a;2024年端午档预售情况&#xff1a;已开启预售&#xff0c;并有望成为该档期的票房冠军备注&#xff1a;据猫眼专业版数据&#xff0c;该片备受期待 《我才不要和你做朋友呢》上映时间&#xff1a;2024年端午档期预…

CSS函数: 实现数据限阈的数字函数

CSS函数中提供了几个比较实用的数字函数&#xff0c;它可以帮助我们实现一定的数学计算功能。常见的数字函数目前提供了五个&#xff1a;calc()、max()、min()和clamp()函数。其基本实现功能如下&#xff1a; calc()&#xff1a;允许在声明 CSS 属性值时执行一些计算。max()&a…

智能推荐算法应用:如何提升淘宝在线扭蛋机用户购物体验

在淘宝的在线扭蛋机平台上&#xff0c;用户的购物体验至关重要。为了提升这一体验&#xff0c;我们引入了智能推荐算法&#xff0c;帮助用户发现他们可能感兴趣的扭蛋产品。这一技术的应用不仅提高了用户的购物效率&#xff0c;还大大增强了用户的购物乐趣。 一、智能推荐算法…

Python语法详解module2(运算符、表达式、流程控制)

目录 一、运算符1. 算术运算符&#xff08;Arithmetic Operators&#xff09;2. 比较运算符&#xff08;Comparison Operators&#xff09;3. 赋值运算符&#xff08;Assignment Operators&#xff09;4. 逻辑运算符&#xff08;Logical Operators&#xff09;5. 位运算符&…

cocos creator3.7版本拖拽事件处理

前言&#xff1a;网上能找到的资料都太落后了&#xff0c;导致哥们用AI去写&#xff0c;全是瞎B写&#xff0c;版本都不对。贴点实际有用的。别老捣鼓你那破convertToNodeSpaceAR或者convertToNodeSpace了。 核心代码 touch.getDeltaX() touch.getDeltaY() 在cocoscreator3…

Python | 刷题日记

1.海伦公式求三角形的面积 area根号下&#xff08;p(p-a)(p-b&#xff09;(p-c)) p是周长的一半 2.随机生成一个整数 import random xrandom.randint(0,9)#随机生成0到9之间的一个数 yeval(input("please input:")) if xy:print("bingo") elif x<y:pri…

sql 查询 不满足 (一个教师编号 的角色 (role =‘2‘or(role=‘1‘and role =‘0‘)) )

sql 查询 不满足 &#xff08;一个教师编号 的角色 &#xff08;role 2’or&#xff08;role1’and role ‘0’&#xff09;&#xff09; &#xff09; 准备 一个 teacher 表 和数据 表 teacher 和数据 -- ---------------------------- -- Table structure for teacher -- …

[Algorithm][动态规划][回文串问题][分割回文串 II][最长回文子序列][让字符串成为回文串的最少插入次数]详细讲解

目录 1.分割回文串 II1.题目链接2.算法原理详解3.代码实现 2.最长回文子序列1.题目链接2.算法原理详解3.代码实现 3.让字符串成为回文串的最少插入次数1.题目链接2.算法原理详解3.代码实现 1.分割回文串 II 1.题目链接 分割回文串 II 2.算法原理详解 思路&#xff1a; 确定状…

现代操作系统(第四版)课后习题-4.文件系统

四、文件系统 1.给出文件/etc/passwd的五种不同的路径名 在Unix和类Unix系统中&#xff0c;/etc/passwd 是系统中所有用户账户信息的标准文件。 绝对路径&#xff1a;直接使用文件的绝对路径来访问。 /etc/passwd使用环境变量&#xff1a;如果设置了环境变量指向某个目录&…

Java中的接口与抽象类:区别与联系

Java中的接口与抽象类&#xff1a;区别与联系 在Java中&#xff0c;interface&#xff08;接口&#xff09;和abstract class&#xff08;抽象类&#xff09;是两种重要的抽象类型&#xff0c;用于定义对象的抽象行为和结构。虽然Java 8之后接口引入了默认方法和静态方法&…