代码随想录算法训练营第十六天| 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,一经查实,立即删除!

相关文章

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 高并发、高可用、分布式、高性能、并…

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

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

Collection与数据结构 顺序表与ArrayList

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

A Novel Negative Sample Generating Method for KnowledgeGraph Embedding

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

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

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

瑞吉外卖实战学习--登录过滤器和判断是否登录过

完善登录功能 1、创建自定义过滤器LoginCheckFiler1.1通过WebFilter创建过滤器1.2 验证是否可以拦截请求1.3 代码 2、在启动类加入注解ServletComponentScan 用来扫描过滤器触发所有的过滤器ServletComponentScan 3、完善过滤器的处理逻辑3.1判断是否需要是要放行的请求3.2判断…

鸿蒙OS应用示例:【数字滚动计时】

实现效果&#xff1a; 代码示例&#xff1a; RollingText.ets 组件封装 RollingText.ets 组件封装 /*** 滚动文字特效*/ Component export default struct RollingText {private num:numberprivate timerId: number -1State counter: number 0aboutToAppear() {this.timerId…

Git基础(25):Cherry Pick合并指定commit id的提交

文章目录 前言指定commit id合并使用TortoiseGit执行cherry-pick命令 前言 开发中&#xff0c;我们会存在多个分支开发的情况&#xff0c;比如dev&#xff0c;test, prod分支&#xff0c;dev分支在开发新功能&#xff0c;prod作为生产分支已发布。如果某个时候&#xff0c;我们…

3.26C++

定义一个矩形类&#xff08;Rectangle&#xff09;&#xff0c;包含私有成员&#xff1a;长(length)、宽&#xff08;width&#xff09;, 定义成员函数&#xff1a; 设置长度&#xff1a;void set_l(int l) 设置宽度&#xff1a;void set_w(int w) 获取长度&#xff1a;int…

【Linux】线程同步{死锁/线程同步相关接口/由浅入深理解线程同步}

文章目录 1.死锁1.1概念1.2死锁的必要条件 2.线程同步相关接口2.1pthread_cond_init/destroy()2.2int pthread_cond_wait2. 3linux下的条件变量及其作用2.4int pthread_cond_signal/broadcast();2.5Linux下 阻塞和挂起的异同2.6阻塞&#xff0c;挂起&#xff0c;和进程切换的关…

【MySQL】数据库--基础

目录 一、概念&#xff1a; 二、连接数据库[Dos命令] 三、SQL 语句分类 一、概念&#xff1a; MySQL 是一种开源的关系数据库管理系统 (RDBMS)数据库-表的本质仍然是文件 二、连接数据库[Dos命令] mysql -h&#xff1a;mysql服务的主机&#xff08;默认连接到本机服务器&…

轻松掌握:使用 API 接口自动缩短网址的秘诀

在互联网的世界里&#xff0c;网址缩短已经成为了一种时尚和必要。长而复杂的网址不仅难以记忆&#xff0c;还可能让人望而却步。但是&#xff0c;现在有了 API 接口&#xff0c;我们可以轻松地将网址自动缩短&#xff0c;让分享变得更加简单和高效&#xff01;本文将以具体例子…

自增不再简单:深入探索MySQL自增ID的持久化之道

概述 MySQL中的自增特性估计大家或多或少都是用过。一张表中只能由一个自增字段&#xff0c;通常我们会把它设置为主键&#xff0c;但是随着大家系统越来越分布式&#xff0c;为了一些性能和可扩展性问题&#xff0c;大家目前选择更多的都是分布式ID&#xff08;雪花算法、UUI…

【python】Jupyter Notebook 修改默认路径

文章目录 一、修改前&#xff08;一&#xff09;问题&#xff08;二&#xff09;修改前的默认路径 二、修改配置文件、更改路径&#xff08;一&#xff09;找到配置文件并打开&#xff08;二&#xff09;创建目标文件夹、得到新的路径&#xff08;三&#xff09;修改配置文件 三…

大模型时代的向量数据库:原理解析和应用案例

大家好&#xff0c;在人工智能领域&#xff0c;数据处理和加工的需求愈发增加。随着人们深入探索AI高级的应用&#xff0c;如图像识别、语音搜索和推荐引擎等&#xff0c;数据的复杂性也在不断地增加。此时传统的数据库存储方式已不能完全满足需求&#xff0c;向量数据库应运而…

su怎么做展厅模型---模大狮模型网

要在SketchUp中创建展厅模型&#xff0c;你可以按照以下基本步骤进行&#xff1a; 绘制基本结构&#xff1a; 使用SketchUp的绘图工具(线条、矩形、圆形等)来创建展厅的基本结构&#xff0c;包括墙壁、地板和天花板等。确保按照实际尺寸和比例进行绘制。 添加家具和展品&…

【C++教程从0到1入门编程】第十三篇:STL中list类的模拟实现

一、list的模拟实现 #include<iostream> #include<assert.h> #pragma once namespace jyr {template<class T>struct _list_node{_list_node<T>* _next;_list_node<T>* _prev;T _data;_list_node(const T& val T()):_next(nullptr), _prev(…

Zabbix 配置使用

目录 配置流程 添加组机组 添加模板 添加主机 配置图形 配置大屏 Monitoring 配置地图 最新数据 故障 使用IT服务 使用报表 资产管理 全局搜索 导入导出 用户权限 用户组权限 用户 匿名用户 调试模式 与 LDAP 对接 维护模式 故障确认 批量更新 配置流程…

【办公类-21-10】三级育婴师 视频转文字docx(等线小五单倍行距),批量改成“宋体小四、1.5倍行距、蓝色字体、去掉五分钟”

作品展示 背景需求 今天将最后3个育婴师操作视频做整理 第1步&#xff1a;视频MP4转MP3 【办公类-40-01】20240311 用Python将MP4转MP3提取音频 &#xff08;家长会系列一&#xff09;-CSDN博客文章浏览阅读393次&#xff0c;点赞9次&#xff0c;收藏6次。【办公类-40-01】20…