算法练习第18天|111.二叉树的最小深度

111.二叉树的最小深度

111. 二叉树的最小深度 - 力扣(LeetCode)icon-default.png?t=N7T8https://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

题目分析:

按照上一篇文章中所写的求二叉树的最大深度对的解法,这道题应该不难。

后序递归解法:

采用比较容易理解的后续递归及递归三部曲可以得到如下代码:

1. 递归第一步:确定递归函数的参数和返回值

参数为要传入的二叉树根节点,返回的是int类型的深度。

代码如下:

int getDepth(TreeNode* node)

2. 递归第二步:确定终止条件 

终止条件也是遇到空节点返回0,表示当前节点的高度为0。

代码如下:

if (node == NULL) return 0;

3. 递归第三步:确定单层递归的逻辑 

这块和求最大深度可就不一样了,一些同学可能会写如下代码:

int leftDepth = getDepth(node->left);
int rightDepth = getDepth(node->right);
int result = 1 + min(leftDepth, rightDepth);  //关键问题出在这一步
return result;

 对于左右子树均存在时,上述代码没有问题。但是当子树不存在时,由前两步getDepth()得到的深度会存在0,如下图所示:

回顾一下题目所给的最小深度的定义:

最小深度是从根节点到最近叶子节点的最短路径上的节点数量。注意是到最近的叶子节点,即没有左右子节点的叶子节点。

那么刚刚所写的单层递归逻辑的代码中是没有判断是否为叶子节点的功能的:

int leftDepth = getDepth(node->left);
int rightDepth = getDepth(node->right);
int result = 1 + min(leftDepth, rightDepth);  //关键问题出在这一步
return result;

补救措施就是在得到leftDepth和rightDepth之后,要判断一下左右子树的存在情况。

  • 如果左子树为空,右子树不为空,说明最小深度是 1 + 右子树的深度。
  • 反之,右子树为空,左子树不为空,最小深度是 1 + 左子树的深度。
  • 最后如果左右子树都不为空,返回左右子树深度最小值 + 1 。

 代码如下:

int leftDepth = getDepth(node->left);           // 左
int rightDepth = getDepth(node->right);         // 右
                                                // 中
// 当一个左子树为空,右不为空,这时并不是最低点
if (node->left == NULL && node->right != NULL) { 
    return 1 + rightDepth;
}   
// 当一个右子树为空,左不为空,这时并不是最低点
if (node->left != NULL && node->right == NULL) { 
    return 1 + leftDepth;
}
int result = 1 + min(leftDepth, rightDepth);
return result;

整体代码如下:

class Solution {
public:
    int getDepth(TreeNode* node) {
        if (node == NULL) return 0;
        int leftDepth = getDepth(node->left);           // 左
        int rightDepth = getDepth(node->right);         // 右
                                                        // 中
        // 当一个左子树为空,右不为空,这时并不是最低点
        if (node->left == NULL && node->right != NULL) { 
            return 1 + rightDepth;
        }   
        // 当一个右子树为空,左不为空,这时并不是最低点
        if (node->left != NULL && node->right == NULL) { 
            return 1 + leftDepth;
        }
        int result = 1 + min(leftDepth, rightDepth);
        return result;
    }

    int minDepth(TreeNode* root) {
        return getDepth(root);
    }
};

上述代码是写了一个专门的getDepth函数进行递归,也可以不写,直接在力扣提供的函数minDepth上进行递归,如下所示:

class Solution {
public:
    int minDepth(TreeNode* root) {
        if(root == nullptr) return 0;
        //左
        int leftDepth = minDepth(root->left);
        //右
        int rightDepth = minDepth(root->right);

        if(root->left == nullptr && root->right != nullptr) 
            return rightDepth + 1;

        if(root->left != nullptr && root->right == nullptr)
            return leftDepth + 1;
        
        int result = min(leftDepth, rightDepth) + 1;
        return result;

    }
};

注意,上面两个写法中的result = min(leftDepth, rightDepth) + 1; 这行代码即包含了左右子树均存在的情况,还包含了左右子树均不存在的情况。均存在时,就去最小值加1;均不存在时,leftDepth和rightDepth均为0,那么算上此时两个空子树的父节点那一层,result做了加1操作(可以想象整个二叉树只有一个根节点组成时的情形)。

所以,代码里的+1操作,就是在后序遍历中处理完了左右子树后,开始考虑子树对应的根节点计算深度(左右中)。

层序遍历解法:

在二叉树层序遍历文章的基础上来实现最小深度的求解,关键一点就是第一次遍历到某个节点的左右孩子都为空的时候,说明遍历到最低点了,此时应该直接return depth,程序运行结束。如果其中一个孩子不为空则不是最低点。

代码如下:

class Solution {
public:
    //层序遍历
    int minDepth(TreeNode* root) {
        if(root == nullptr) return 0;
        queue<TreeNode*> que;
        que.push(root);
        int depth = 0;
        while(!que.empty())
        {
            int size = que.size();
            depth++;
            for(int i = 0; i < size; ++i)
            {
                TreeNode*  node = que.front();
                que.pop();
                //最低处判断放这里也行
                // if(node->left == nullptr && node->right == nullptr)
                //     return depth;
                if(node->left) que.push(node->left);
                if(node->right) que.push(node->right);
                if(node->left == nullptr && node->right == nullptr)
                    return depth;               
            }
        }
        return depth;

    }
};

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

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

相关文章

bootstrap-select 搜索过滤输入中文问题,前2个字母输入转成空格

bootstrap是v3.3.7的 v1.6.3版本的bootstrap-select,注释以下2行 //that.$menu.find(li).filter(:visible:not(.divider)).eq(0).addClass(active).find(a).focus(); // $(this).focus();

Dryad Girl Fawnia

一个可爱的Dryad Girl Fawnia的三维模型。她有ARKit混合形状,人形装备,多种颜色可供选择。她将是一个完美的角色,幻想或装扮游戏。 🔥 Dryad Girl | Fawnia 一个可爱的Dryad Girl Fawnia的三维模型。她有ARKit混合形状,人形装备,多种颜色可供选择。她将是一个完美的角色…

工业控制(ICS)---COTP协议

COTP 可以理解为基于TCP的工控TCP&#xff0c;主要有五种类型&#xff1a; CR Connect Request (0x0e)——握手&#xff0c;发送方发送 CC Connect Confirm (0x0d)——握手&#xff0c;接收方发送 DT Data (0x0f)——传正常数据 UD User Data (0x04)——少见&#xff0c;传…

mysql多表查询时与子表的关系

力扣题目链接 如果最后一句代码不把a与两个表都进行连接的话会出现结果重复的问题

全网短剧搜索源码+短剧API接口 短剧下载 热门短剧 全开源可二开

内容目录 一、详细介绍二、效果展示1.部分代码2.效果图展示 三、学习资料下载 一、详细介绍 pc端h5手机端全网短剧搜索前端源码分享 内含7000短剧资源(不支持在线播放&#xff09; 搜索API接口&#xff1a;文件内查看 全部短剧API接口&#xff1a;文件内查看 每日更新API接…

《黑神话:悟空》现已正式上架PS商城,今晚或有大动作

关于《黑神话&#xff1a;悟空》的消息可谓是喜闻乐见&#xff01;今天晚上19:10可能会有相关游戏内容放出&#xff0c;让人非常期待。而海信电视推出的《黑神话&#xff1a;悟空》专属画质模式&#xff0c;让玩家可以享受到更加细腻的游戏画面。 此外&#xff0c;海信和《黑神…

Spring Boot - 利用MDC(Mapped Diagnostic Context)实现轻量级同步/异步日志追踪

文章目录 Pre什么是MDC&#xff08;Mapped Diagnostic Context&#xff09;Slf4j 和 MDC基础工程工程结构POMlogback-spring.xmlapplication.yml同步方式方式一&#xff1a; 拦截器自定义日志拦截器添加拦截器 方式二&#xff1a; 自定义注解 AOP自定义注解 TraceLog切面 测试…

Oracle 数据库全表扫描的4种优化方法(DB)

全表扫描的工作是扫描高水位一下所有的数据块。 这里就有一个问题&#xff0c;什么是高水位线。高水位的标志存在表头。 该数据块以后都是崭新未格式化的数据块&#xff0c;高水位的目的有二。它是全表扫描的 终点&#xff0c;并行插入的起点&#xff01; 优化全表扫描的办法有…

1688官方API商品数据采集接口|阿里巴巴中国站获得1688商品详情 API 返回值说明

随着全球经济一体化和电子商务的快速发展&#xff0c;网络购物的需求日益增加。不断涌现的电商企业使得行业的竞争情况愈演愈烈。在这种情况下&#xff0c;企业不仅要加大经营力度&#xff0c;还要在自己的基础设施和技术上持续投入&#xff0c;才能更好的适应市场和消费习惯。…

李沐-19 卷积层【动手学深度学习v2】

记录下关于权重下标变换的理解&#xff1a; 从原来的Wi,j到Wi,j,k,l是从二维到四维的过程&#xff0c;如下图所示 对全连接层使用平移不变性(如&#xff1a;卷积核在移动过程是不变的)和局部性&#xff08;如&#xff1a;卷积核有一定大小&#xff09;得到卷积层&#xff0c;这…

JetBrains2024来袭

JetBrains2024来袭&#xff0c;激活包含在内的编程IDE&#xff0c;其中AppCode已下架&#xff0c;Aqua&#xff0c;RustRover不支持本地激活需要关联帐号。 Tap&#xff1a;激活稳定可靠&#xff0c;支持Windows&#xff0c;macOS&#xff0c;Linux客户端。

Linux知识

基础 Linux系统的组成 Linux内核、Linux⽂件系统、Linux shell、Linux应⽤程序。 Linux的开机启动过程 u-boot是⼀款常⽤的开源Bootloader&#xff0c;它的启动顺序如下&#xff1a; CPU上电后&#xff0c;⾸先执⾏boot ROM&#xff08;引导ROM&#xff09;代码。boot ROM…

历史上9个不可思议的巧合

历史是神奇的&#xff0c;在历史上&#xff0c;有些明明没道理的事情&#xff0c;却偏偏就这样发生了&#xff0c;明明不相关的事情&#xff0c;也偏偏就有那么一个巧合对应了。 1、国际通用的公元元年正好对应着西汉刘衎的元始元年。而这个年号正是王莽所定&#xff0c;而王莽…

软考130-上午题-【软件工程】-系统维护

一、系统维护概述 软件维护是软件生命周期中的最后一个阶段&#xff0c;处于系统投入生产性运行以后的时期中&#xff0c;因此不属于系统开发过程。 软件维护是在软件已经交付使用之后为了改正错误或满足新的需求而修改软件的过程&#xff0c;即软件在交付使用后对软件所做的一…

Python | Leetcode Python题解之第25题K个一组翻转链表

题目&#xff1a; 题解&#xff1a; class Solution:# 翻转一个子链表&#xff0c;并且返回新的头与尾def reverse(self, head: ListNode, tail: ListNode):prev tail.nextp headwhile prev ! tail:nex p.nextp.next prevprev pp nexreturn tail, headdef reverseKGroup…

JQuery(四)---【使用JQuery实现动画效果】

目录 前言 一.隐藏和显示 1.1使用方法 1.2案例演示(1) 1.3隐藏/显示效果一键切换 二.淡入淡出效果 2.1使用方法 2.2案例演示(fadeIn) 2.3案例演示(fadeOut) 2.4案例演示(fadeToggle) 2.5案例演示(fadeTo) 三.滑动 3.1使用方法 3.2案例演示(slideDown) 3.3案例演示…

SpringBoot中使用Jackson序列化返回

SpringBoot中使用Jackson序列化返回 在Spring Boot应用中&#xff0c;使用Jackson库来处理JSON的序列化和反序列化是一种常见的做法。Jackson是一个高效的JSON处理器&#xff0c;广泛用于Java环境中&#xff0c;尤其是在与Spring框架集成时。本文将详细介绍如何在Spring Boot中…

预付费电表管理系统

预付费电表管理系统是一种现代化的电力管理系统&#xff0c;它将信息化技术和电力系统紧密结合&#xff0c;旨在提供更加高效、便捷的电力使用与管理方式。该系统能够有效地解决传统计费方法中存在的延迟计费、欠费风险等问题&#xff0c;通过预付费的方式&#xff0c;既保证了…

【动态规划】【01背包 尽量装满背包】Leetcode 1049. 最后一块石头的重量 II

【动态规划】【01背包 尽量装满背包】Leetcode 1049. 最后一块石头的重量 II 解法 ---------------&#x1f388;&#x1f388;题目链接&#x1f388;&#x1f388;------------------- 解法 &#x1f612;: 我的代码实现> 动规五部曲 ✒️确定dp数组以及下标的含义 d…

[数据结构]——二叉树链式结构的实现

目录 1. 前置说明 2. 二叉树的遍历 1. 前序、中序以及后序遍历 1.前序遍历递归 1.图解&#xff1a;​编辑 2.代码 2.中序遍历递归 3.后序遍历递归 3. 节点个数以及高度等 1.二叉树节点个数 2.叶子节点个数 3.树的高度 4.K层节点个数 5.二叉树查找值为x的节点是否存在…