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

本文主要是介绍代码随想录算法训练营第十三天 | 104.二叉树的最大深度、559.N叉树的最大深度、111.二叉树的最小深度、222.完全二叉树的节点个数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

文章目录

  • 代码随想录算法训练营第十三天 | 104.二叉树的最大深度、559.N叉树的最大深度、111.二叉树的最小深度、222.完全二叉树的节点个数
    • 1 LeetCode 104.二叉树的最大深度
    • 2 LeetCode 559.N叉树的最大深度
    • 3 LeetCode 111.二叉树的最小深度
    • 4 LeetCode 222.完全二叉树的节点个数
      • 4.1 普通遍历
      • 4.2 利用完全二叉树的特性优化遍历

1 LeetCode 104.二叉树的最大深度

题目链接:https://leetcode.cn/problems/maximum-depth-of-binary-tree/description/

给定一个二叉树 root ,返回其最大深度。

二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。

示例 1:

在这里插入图片描述

输入:root = [3,9,20,null,null,15,7]
输出:3

示例 2:

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

提示:

  • 树中节点的数量在 [0, 104] 区间内。
  • -100 <= Node.val <= 100

这道题目其实也就是求二叉树的高度,这种类型的题目408考试也可能会出,需要注意一下,这道题目我们就考虑用递归法做,那么前序遍历和后序遍历都可以实现。

首先后序遍历的代码很简单。

(1)Python版本代码

# 后序遍历
class Solution:def maxDepth(self, root: Optional[TreeNode]) -> int:if not root:return 0L = self.maxDepth(root.left)R = self.maxDepth(root.right)if L > R:return L + 1else:return R + 1

我们只需要记录最高的一棵子树,然后加上根节点,就可以知道整颗二叉树的最大高度(深度)。

下面是前序遍历,我们需要额外的定义一个函数,用于递归传参。

# 前序遍历
class Solution:def maxDepth(self, root: Optional[TreeNode]) -> int:return self.getHeight(root, 0)def getHeight(self, root, cur_depth):if not root:return cur_depthL = self.getHeight(root.left, cur_depth + 1)R = self.getHeight(root.right, cur_depth + 1)return max(L, R)

(2)C++版本代码

// 后序遍历
class Solution {
public:int maxDepth(TreeNode* root) {if (!root) {return 0;}int L = maxDepth(root->left);int R = maxDepth(root->right);return std::max(L, R) + 1;}
};// 前序遍历
class Solution {
public:int maxDepth(TreeNode* root) {return getHeight(root, 0);}int getHeight(TreeNode* root, int cur_depth) {if (!root) {return cur_depth;}int L = getHeight(root->left, cur_depth + 1);int R = getHeight(root->right, cur_depth + 1);return std::max(L, R);}
};

2 LeetCode 559.N叉树的最大深度

题目链接:https://leetcode.cn/problems/maximum-depth-of-n-ary-tree/description/

给定一个 N 叉树,找到其最大深度。

最大深度是指从根节点到最远叶子节点的最长路径上的节点总数。

N 叉树输入按层序遍历序列化表示,每组子节点由空值分隔(请参见示例)。

示例 1:

img

输入:root = [1,null,3,2,4,null,5,6]
输出:3

示例 2:

img

输入:root = [1,null,2,3,4,5,null,null,6,7,null,8,null,9,10,null,null,11,null,12,null,13,null,null,14]
输出:5

提示:

  • 树的深度不会超过 1000
  • 树的节点数目位于 [0, 104] 之间。

这道题目和上一道题目的思路一样,只不过从遍历一个节点的左右孩子变成了遍历一个节点的孩子,下面我们直接给出后序遍历的代码。

(1)Python版本代码

"""
# Definition for a Node.
class Node:def __init__(self, val=None, children=None):self.val = valself.children = children
"""class Solution:def maxDepth(self, root: 'Node') -> int:if not root:return 0depth = 0for i in range(len(root.children)):depth = max(depth, self.maxDepth(root.children[i]))return depth + 1

(2)C++版本代码

/*
// Definition for a Node.
class Node {
public:int val;vector<Node*> children;Node() {}Node(int _val) {val = _val;}Node(int _val, vector<Node*> _children) {val = _val;children = _children;}
};
*/class Solution {
public:int maxDepth(Node* root) {if (root == NULL) return 0;int depth = 0;for (int i = 0; i < root->children.size(); i++) {depth = max (depth, maxDepth(root->children[i]));}return depth + 1;}
};

3 LeetCode 111.二叉树的最小深度

题目链接:https://leetcode.cn/problems/minimum-depth-of-binary-tree/description/

给定一个二叉树,找出其最小深度。

最小深度是从根节点到最近叶子节点的最短路径上的节点数量。

**说明:**叶子节点是指没有子节点的节点。

示例 1:

在这里插入图片描述

输入:root = [3,9,20,null,null,15,7]
输出:2

示例 2:

输入:root = [2,null,3,null,4,null,5,null,6]
输出:5

提示:

  • 树中节点数的范围在 [0, 105]
  • -1000 <= Node.val <= 1000

上面两题是求二叉树的最大深度,本题刚好相反,有相同的地方,也有不同的地方,首先递归终止条件肯定一样,然后也是需要判断根节点左右子树是否为空,如果仅左子树为空,那么二叉树就只剩下右子树,也就类似于链表的结构,此时最小深度也就是树高,同理仅右子树为空也是这样,唯一不同的是如果左右子树都不为空,那么我们就需要返回深度更小的高度,这样我们就可以得到二叉树的最小深度了。

下面我们来写出对应的代码,从上面分析我们不难看出使用的是递归法中的后序遍历。

(1)Python版本代码

# Definition for a binary tree node.
# class TreeNode:
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution:def minDepth(self, root: Optional[TreeNode]) -> int:if not root:return 0if not root.left:  # 左子树为空return self.minDepth(root.right) + 1    # 返回右子树的最小深度 + 1if not root.right:  # 右子树为空return self.minDepth(root.left) + 1    # 返回左子树的最小深度 + 1return min(self.minDepth(root.left), self.minDepth(root.right)) + 1    # 左右子树都不为空,返回左右子树最小深度的较小值 + 1

(2)C++版本代码

#include <iostream>
#include <algorithm>/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode() : val(0), left(nullptr), right(nullptr) {}*     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}*     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}* };*/class Solution {
public:int minDepth(TreeNode* root) {if (!root) {return 0;}if (!root->left) {return minDepth(root->right) + 1;}if (!root->right) {return minDepth(root->left) + 1;}return std::min(minDepth(root->left), minDepth(root->right)) + 1;}
};

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

题目链接:https://leetcode.cn/problems/count-complete-tree-nodes/description/

给你一棵 完全二叉树 的根节点 root ,求出该树的节点个数。

完全二叉树 的定义如下:在完全二叉树中,除了最底层节点可能没填满外,其余每层节点数都达到最大值,并且最下面一层的节点都集中在该层最左边的若干位置。若最底层为第 h 层,则该层包含 1~ 2h 个节点。

示例 1:

在这里插入图片描述

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

示例 2:

输入:root = []
输出:0

示例 3:

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

提示:

  • 树中节点的数目范围是[0, 5 * 104]
  • 0 <= Node.val <= 5 * 104
  • 题目数据保证输入的树是 完全二叉树

**进阶:**遍历树来统计节点是一种时间复杂度为 O(n) 的简单解决方案。你可以设计一个更快的算法吗?

这道题目最好实现的思路就是把完全二叉树当作一棵普通的二叉树来计算,这样思路会非常的清晰,代码也非常的简短,时间复杂度也就是遍历二叉树所花费的时间:O(n),我们最好采用后序遍历的方法来实现,相比于前序和中序代码更好写,我们只需要分别统计左子树和右子树的节点数量,最后加上根节点即可。

4.1 普通遍历

(1)Python版本代码

# Definition for a binary tree node.
# class TreeNode:
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution:def countNodes(self, root: Optional[TreeNode]) -> int:if not root:return 0leftNums = self.countNodes(root.left)rightNums = self.countNodes(root.right)result = leftNums + rightNums + 1return result

(2)C++版本代码

/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode() : val(0), left(nullptr), right(nullptr) {}*     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}*     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}* };*/
class Solution {
public:int countNodes(TreeNode* root) {if (root == NULL)   return 0;int leftNums = countNodes(root->left);int rightNums = countNodes(root->right);int reslut = leftNums + rightNums + 1;return reslut;}
};

4.2 利用完全二叉树的特性优化遍历

题目给出的是一棵完全二叉树,所以我们还可以用完全二叉树的特性进行剪枝操作,比如,如果一棵子树,它的左右子树如果高度一样,那么它就是一棵满二叉树,因为如果中间叶子节点缺失那么它就不满足题目条件就不是一棵完全二叉树了(完全二叉树的介绍可以看前几期的博客,另外二叉树的各种类型及其特性是需要牢记于心的),如果满足左右子树高度相等的话,这个时候我们就可以直接使用对应计算满二叉树节点数的公式( 2 h − 1 , h 为二叉树的高度 2^h-1,h为二叉树的高度 2h1,h为二叉树的高度)计算其节点总数,最后再相加起来即可。

下面是实现的代码。

(1)Python版本代码

# Definition for a binary tree node.
# class TreeNode:
#     def __init__(self, val=0, left=None, right=None):
#         self.val = val
#         self.left = left
#         self.right = right
class Solution:def countNodes(self, root: Optional[TreeNode]) -> int:if not root:return 0left = root.leftright = root.rightleftHight = 0rightHight = 0while left:left = left.leftleftHight += 1while right:right = right.rightrightHight += 1if leftHight == rightHight:return (2 << leftHight) - 1leftNums = self.countNodes(root.left)rightNums = self.countNodes(root.right)result = leftNums + rightNums + 1return result

(2)C++版本代码

class Solution {
public:int countNodes(TreeNode* root) {if (root == nullptr) return 0;TreeNode* left = root->left;TreeNode* right = root->right;int leftDepth = 0, rightDepth = 0; // 这里初始为0是有目的的,为了下面求指数方便while (left) {  // 求左子树深度left = left->left;leftDepth++;}while (right) { // 求右子树深度right = right->right;rightDepth++;}if (leftDepth == rightDepth) {return (2 << leftDepth) - 1; // 注意(2<<1) 相当于2^2,所以leftDepth初始为0}return countNodes(root->left) + countNodes(root->right) + 1;}
};

这篇关于代码随想录算法训练营第十三天 | 104.二叉树的最大深度、559.N叉树的最大深度、111.二叉树的最小深度、222.完全二叉树的节点个数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/669179

相关文章

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

nginx-rtmp-module模块实现视频点播的示例代码

《nginx-rtmp-module模块实现视频点播的示例代码》本文主要介绍了nginx-rtmp-module模块实现视频点播,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录预置条件Nginx点播基本配置点播远程文件指定多个播放位置参考预置条件配置点播服务器 192.