代码随想录算法训练营第十六天| 104.二叉树的最大深度、559.n叉树的最大深度、111.二叉树的最小深度、222.完全二叉树的节点个数

系列文章目录


目录

  • 系列文章目录
  • 104.二叉树的最大深度
    • ①递归法
      • 直接法(求深度,前序遍历)
      • 间接法(求高度,后序遍历)
    • ②迭代法(层序遍历中有)
  • 559.n叉树的最大深度
    • ①递归法
      • 间接法(后序遍历求高度)
      • 直接法(前序遍历求深度)
    • ②迭代法(层序遍历)
  • 111.二叉树的最小深度
    • ①递归法
      • 后序遍历求高度
      • 前序遍历求高深度
    • ②迭代法
      • 层序遍历
  • 222.完全二叉树的节点个数
    • 按照普通二叉树的规则
      • ①递归法(后序遍历)
      • ②迭代法
        • 层序遍历
        • 后序遍历(统一迭代)
    • 针对完全二叉树的性质
      • 递归法


104.二叉树的最大深度

①递归法

直接法(求深度,前序遍历)

(1)直接求深度,用前序遍历从上往下遍历,子节点的深度 = 父节点的深度 + 1

(2)递归三部曲:

  1. 确定递归参数和返回值:传入当前节点和父节点的深度。由于是前序遍历,先获取的当前节点的深度,再获取左右节点的深度,此时左右节点的深度返回没有意义。因此,参数为TreeNode类的当前节点,int型的父节点深度,返回值为void
  2. 确定终止条件:当前节点为null时,表示到达了树的底部,计算深度没有意义,直接返回即可。
  3. 确定单层递归逻辑:每次向下递归时,深度 deep 都会增加 1,表示遍历到了下一层。将中节点的深度与当前最大深度maxnum比较,更新最大深度。对当前节点的左右子节点进行递归调用。
//递归法(前序,直接法,求深度)
class Solution {//定义最大深度int maxnum = 0;public int maxDepth(TreeNode root) {getdepth(root, 0);return maxnum;}public void getdepth(TreeNode node, int deep) {//确定终止条件if (node == null) {return;}deep++;maxnum = maxnum < deep ? deep : maxnum;getdepth(node.left, deep);getdepth(node.right, deep);}
}

间接法(求高度,后序遍历)

(1)二叉树的最大深度 = 根节点的最大高度,父节点的最大高度 = 左右子节点高度的最大值 + 1(需要先访问左右节点,再访问中节点,所以对应的是后序遍历)
(2)递归三部曲:

  1. 确定递归参数和返回值:目的是获取当前节点的高度,输入的是TreeNode类型的节点,输出int型高度。
  2. 确定终止条件:如果当前节点为null,则返回高度为0(注意:叶子节点的高度为1)。
  3. 确定单层递归逻辑:获取当前节点的高度,即要获取左右子节点的最大高度再+1
class Solution {public int maxDepth(TreeNode root) {if (root == null) {return 0;}int leftDepth = maxDepth(root.left);int rightDepth = maxDepth(root.right);return Math.max(leftDepth, rightDepth) + 1;}
}

②迭代法(层序遍历中有)

class Solution {public int maxDepth(TreeNode root) {//迭代法Queue<TreeNode> que = new LinkedList<>();if (root == null) {return 0;}que.offer(root);int deep = 0;while (!que.isEmpty()) {int len = que.size();deep++;// 记录深度while (len-- > 0) {TreeNode tempNode = que.poll();if(tempNode.left!=null)que.offer(tempNode.left);if(tempNode.right!=null)que.offer(tempNode.right);}}return deep;}
}

559.n叉树的最大深度

①递归法

间接法(后序遍历求高度)

确定单层逻辑时,在循环遍历该节点的子节点时获取所有子节点的高度的最大值,再在循环后+1作为当前节点的高度。

class Solution {public int maxDepth(Node root) {//递归法(后序)if (root == null) return 0;int maxnum = 0;for (Node child : root.children) {maxnum=maxnum < maxDepth(child) ? maxDepth(child) : maxnum;}return maxnum + 1;//中节点}
}

直接法(前序遍历求深度)

class Solution {int maxNum=0;public int maxDepth(Node root) {getDepth(root,0);return maxNum;}public void getDepth(Node node,int deep){if(node==null)return;deep++;maxNum=Math.max(maxNum,deep);for(Node child:node.children){getDepth(child,deep);}}
}

②迭代法(层序遍历)

注:在遍历N个子节点时注意代码的健壮性,要考虑当前节点是否有子节点链表以及链表中的子节点是否为null,不为空时才加入队列中。

List<Node> list = node.children;//代码健壮性if(!list.isEmpty()) {//子节点链表不为空for (Node child : node.children) {if(child!=null){//子节点不为空才加入队列que.offer(child);}}}

完整代码如下:

class Solution {public int maxDepth(Node root) {if (root == null) {return 0;}Queue<Node> que = new LinkedList<>();que.offer(root);int deep = 0;while (!que.isEmpty()) {int len = que.size();deep++;while (len-- > 0) {Node node = que.poll();List<Node> list = node.children;//代码健壮性if(!list.isEmpty()) {//子节点链表不为空for (Node child : node.children) {if(child!=null){//子节点不为空才加入队列que.offer(child);}}}}}return deep;}
}

111.二叉树的最小深度

①递归法

后序遍历求高度

最小深度 = 根节点的最小高度
父节点的最小高度 = 左右节点的高度最小值 + 1

class Solution {public int minDepth(TreeNode root) {if (root == null) {return 0;}int leftDepth = minDepth(root.left); // 左int rightDepth = minDepth(root.right);// 右// 中// 当一个左子树为空,右不为空,这时并不是最低点if(root.left==null&&root.right!=null){return 1+rightDepth;}// 当一个右子树为空,左不为空,这时并不是最低点if(root.left!=null&&root.right==null){return 1+leftDepth;}return Math.min(leftDepth, rightDepth) + 1;}
}

前序遍历求高深度

class Solution {int minNum=Integer.MAX_VALUE;public int minDepth(TreeNode root) {if(root==null){return 0;}getDepth(root,0);return minNum;}public void getDepth(TreeNode node,int deep){// 函数递归终止条件if(node==null)return;deep++;//中,处理逻辑:判断是不是叶子结点if(node.left==null&&node.right==null){minNum=Math.min(minNum,deep);}getDepth(node.left,deep);// 左getDepth(node.right,deep);// 右}
}

②迭代法

层序遍历

class Solution {public int minDepth(TreeNode root) {Queue<TreeNode> que = new LinkedList<>();if (root == null) {return 0;}que.offer(root);int deep = 0;while (!que.isEmpty()) {int len = que.size();deep++;while (len-- > 0) {TreeNode tempNode = que.poll();//如果当前节点的左右孩子都为空,直接返回最小深度if (tempNode.left == null && tempNode.right == null) {return deep;}if (tempNode.left != null) que.offer(tempNode.left);if (tempNode.right != null) que.offer(tempNode.right);}}return deep;}
}

222.完全二叉树的节点个数

按照普通二叉树的规则

①递归法(后序遍历)

class Solution {public int countNodes(TreeNode root) {if (root == null) return 0;int leftNum = countNodes(root.left);//左int rightNum = countNodes(root.right);//右return leftNum + rightNum + 1;//中,左子树的数量+右子树的数量+本身节点}
}

②迭代法

层序遍历
import java.util.LinkedList;
import java.util.Queue;
class Solution {public int countNodes(TreeNode root) {Queue<TreeNode> que = new LinkedList<>();if (root == null) return 0;que.offer(root);int num=0;while(!que.isEmpty()){int len = que.size();while(len-->0){TreeNode node = que.poll();num++;if(node.left!=null)que.offer(node.left);if(node.right!=null)que.offer(node.right);}}return num;}
}
后序遍历(统一迭代)

注:用LinkedList作为栈使用是,只有Deque接口有push方法,Queue接口是单端队列,无法在队列头加入元素!

import java.util.Deque;
import java.util.LinkedList;
class Solution {public int countNodes(TreeNode root) {Deque<TreeNode> que = new LinkedList<>();int num = 0;if (root == null) return num;que.push(root);while (!que.isEmpty()) {TreeNode node = que.peek();if (node != null) {que.poll();que.push(node);//中que.push(null);if (node.right != null) {que.push(node.right);//右}if (node.left != null) {que.push(node.left);//左}}else {//处理节点元素que.poll();que.poll();//再次弹出元素num++;}}return num;}
}

这几种方法都相当于把每个节点都遍历了一遍,所以时间复杂度是O(n)
在这里插入图片描述

针对完全二叉树的性质

判断满二叉树的方法:向左遍历的深度 = 向右遍历的深度,则该树为满二叉树

递归法

(1)递归三部曲:

  1. 确定递归参数和返回值:当前节点root是递归参数,节点数是返回值。
  2. 确定终止条件:①如果当前节点rootnull,则返回0,意味着此时没有节点。②当该节点所在树为满二叉树时,则根据2^k - 1计算该节点对应子树的所有节点数并返回;
  3. 单层递归的逻辑(可以看出使用后序遍历):如果不是满二叉树,继续遍历左节点和右节点,当前节点对应子树的所有节点数 = 左右子树所有节点之和 + 1

(2)算数运算符(-)的优先级比位运算符(<<)的大,故return (2 << leftDepth) - 1;此处需加上()

class Solution {public int countNodes(TreeNode root) {//确定递归参数和返回值//确定终止条件if (root == null) return 0;int leftDepth = 0;int rightDepth = 0; // 这里初始为0是有目的的,为了下面求指数方便TreeNode left = root.left;TreeNode right = root.right;while (left != null) {// 求左子树深度leftDepth++;left = left.left;}while (right != null) { // 求右子树深度rightDepth++;right = right.right;}if (leftDepth == rightDepth) {return (2 << leftDepth) - 1;// 注意(2<<1) 相当于2^2,所以leftDepth初始为0}//单层递归的逻辑(可以看出使用后序遍历)int leftCount = countNodes(root.left);int rightCount = countNodes(root.right);return leftCount + rightCount + 1;}
}

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

自动化脚本-滑动验证码识别登录

结果演示 滑块验证码登录 测试网站 提供各类型验证码识别验证&#xff0c;主要提供人机识别与验证服务 行为验证4.0-适应型验证码-滑动验证,点选验证,图片验证-极验GeeTest 代码实现 相关前置依赖可参考 自动化脚本-图片验证码识别登陆-CSDN博客 具体实现 # -*- coding: u…

优必选校招软件开发岗笔试Java题解

1、计算某天是星期几 输入一个年月日&#xff0c;计算出这个日期对应的是星期几 注意&#xff1a; 闰年 公元年分除以4不可整除&#xff0c;为平年。 公元年分除以4可整除但除以100不可整除&#xff0c;为闰年。 公元年分除以100可整除但除以400不可整除&#xff0c;为平年。 …

【卡梅德生物】哺乳动物细胞表达系统介绍

哺乳动物细胞表达系统是在哺乳动物细胞中生产重组蛋白的广泛使用的平台。哺乳动物细胞为蛋白表达提供了几个优势&#xff0c;包括进行复杂翻译后修饰的能力、蛋白质的正确折叠、将蛋白质分泌到细胞外环境中的能力等。哺乳动物细胞表达系统广泛用于生物技术和制药工业&#xff0…

canvas画带透明度的直线和涂鸦

提示&#xff1a;canvas画线 文章目录 前言一、带透明度的直线和涂鸦总结 前言 一、带透明度的直线和涂鸦 test.html <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content…

咕炮课堂Java架构师课程

课程介绍 主要针对1到5年及以上工作经验的开发人员&#xff0c;提供互联网行业热门技术的Java架构师专题培训&#xff0c;由业内技术大牛&#xff0c;行业及实战经验丰富的讲师进行技术分享。内容涵盖redis,mongodb,dubbo,zookeeper,kafka 高并发、高可用、分布式、高性能、并…

【面经八股】大模型方向:面试记录(二)

【面经&八股】大模型方向:面试记录(二) 文章目录 【面经&八股】大模型方向:面试记录(二)0. 背景1. 一面(3.25)1.1 自我介绍1.2 科研-项目经历问答1.3 实习经历问答1.4 八股1.5 编程题1.6 反问2. 二面(3.26)2.1 自我介绍2.2 八股文2.3 编程0.

macOS Sonoma 14.4.1(23E224)发布(附黑/白苹果镜像)

系统介绍 黑果魏叔3 月 26 日消息&#xff0c;苹果今日向 Mac 电脑用户推送了 macOS 14.4.1 更新&#xff08;内部版本号&#xff1a;23E224&#xff09;&#xff0c;本次更新距离上次发布隔了 18 天。 根据苹果公司的发布说明&#xff0c;macOS Sonoma 14.4.1 更新修复了一个…

雪里温柔,水边明秀,不及Java 抽象类 和 Object类

本篇会加入个人的所谓‘鱼式疯言’ ❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言 而是理解过并总结出来通俗易懂的大白话, 小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的. &#x1f92d;&#x1f92d;&#x1f92d;可能说的不是那么严谨.但小编初心是能让更多人…

Collection与数据结构 顺序表与ArrayList

1. 线性表 线性表&#xff08;linear list&#xff09;是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构&#xff0c;常见的线性表&#xff1a;顺序表、链表、栈、队列… 线性表在逻辑上是线性结构&#xff0c;也就说是连续的一条直线。但是在…

rtt的io设备框架面向对象学习-内部调用流程

IO设备管理层的对象接口一般调用流程如下 rt_device_find->rt_device_open->rt_device_read->rt_device_write->rt_device_close 再加上各个设备特有使用方法。参见官方文档查看各个设备的是使用方法。 rt_device_find&#xff08;name&#xff09;是在对象容器里…

C++语法|C++八股|内存分区、内存对齐、野指针和悬浮指针

文章目录 内存分区堆(heap)和栈(stack)的区别new和malloc的区别delete和free有什么区别 野指针导致野指针的原因如何避免野指针野指针和悬浮指针 内存对齐什么是内存对其为什么要内存对齐内存对其的规则 内存分区 从高地址到低地址依次是&#xff1a; 栈(stack)区&#xff1a…

Word字号与磅值与行距

文章目录 简介字号、磅值与行距之间的对应关系 简介 在论文等文字资料进行排版的过程中&#xff0c;悉知字号对应的磅值&#xff0c;以及行距之间的关系非常有帮助。本篇博文简单介绍一下排版领域中有关中文字号&#xff08;三号、小四、五号、小五等&#xff09;与磅值&#…

Redis 教程系列之Redis 分区(十)

Redis 分区 分区是分割数据到多个Redis实例的处理过程&#xff0c;因此每个实例只保存key的一个子集。 分区的优势 通过利用多台计算机内存的和值&#xff0c;允许我们构造更大的数据库。通过多核和多台计算机&#xff0c;允许我们扩展计算能力&#xff1b;通过多台计算机和…

A Novel Negative Sample Generating Method for KnowledgeGraph Embedding

摘要 为了有效地提取知识图中的关系和原因&#xff0c;将实体和关系编码到一个连续的低维语义空间中。在负样本生成阶段&#xff0c;大多数知识图嵌入方法更注重替换头或尾实体以提高训练效率&#xff0c;很少替换关系。这些负样本生成方法对关系预测的贡献不大。本文提出了一…

vue项目在本地源码方式启动和打包之后在nginx中代理有什么不同

Vue项目在本地源码方式启动和打包之后在Nginx中代理的主要区别在于开发环境与生产环境的配置、性能优化、安全性和部署流程等方面。以下是一些具体的差异点&#xff1a; 开发环境与生产环境&#xff1a; 本地源码启动通常是在开发环境中&#xff0c;使用Vue CLI的vue-cli-servi…

有效沟通(业务分析关键技能)

背景 业务分析有一筐子技能&#xff0c;如果让我选一个最重要的&#xff0c;那么就是如何有效沟通&#xff0c;这也可以从项目的另外一个角度思考&#xff0c;项目如何保障能安全着陆&#xff0c;那就是别走偏了。 走偏了&#xff0c;大多数就是对项目理解不清楚&#xff0c;…

智能网络运维:领航数字时代,实现网络管理极致效能

在当今高度信息化的时代&#xff0c;网络已经成为企业运营不可或缺的一部分。网络设备的稳定运行、数据传输的畅通无阻&#xff0c;都直接关系到企业的正常运营和业务发展。因此&#xff0c;高效、智能的网络运维管理显得尤为重要。本文将重点介绍智能网络运维中的几个关键模块…

更新对象的部分输入参数

更新对象的部分输入参数 代码 def update_state(self, **kwargs):# 更新指定的状态参数&#xff0c;保持其他参数不变for key, value in kwargs.items():if hasattr(self, key):setattr(self, key, value)怎么理解解释 用于更新对象的状态参数。这个方法使用了关键字参数&am…

8、Spring CLI中AI命令指南

AI 命令指南 OpenAI 的 ChatGPT 等大型语言模型为使用 AI 生成代码提供了强大的解决方案。ChatGPT 不仅在 Java 代码上进行了训练,还在 Spring 开源生态系统内的各种项目上进行了训练。这使得 Spring CLI 能够增强应用程序的功能,超出传统教程所能提供的范围。 使用简单的命…

每天学习一会java(第一天)----条件运算符

今天学习的是条件运算符 1.描述&#xff1a; 条件运算符由“?”与 “:” 两个符号组成&#xff0c;必须一起使用&#xff0c;是 JAVA 中唯一的三目&#xff08;三元&#xff09;运算符&#xff0c;需要三个操作数才能进行运算。 条件表达式的一般使用形式为&#xff1a; 表达…