代码随想录算法训练营打卡Day16 | LeetCode104 二叉树的最大深度、LeetCode111 二叉树的最小深度、LeetCode222 完全二叉树的节点个数

摘要

  • 二叉树的深度是从根节点出发,到最远的叶节点的最长路径上的节点数
  • 二叉树的高度是从最远叶节点出发,到根节点的最长路径上的节点数
  • 完全二叉树可以的节点个数和树的高度与深度有明显的规律

LeetCode104 二叉树的深度

104. 二叉树的最大深度 - 力扣(Leetcode)

  • 关键概念
    • 二叉树节点的深度:指从根节点到该节点的最长简单路径边的条数或者节点数(取决于深度从0开始还是从1开始)
    • 二叉树节点的高度:指从该节点到叶子节点的最长简单路径边的条数或者节点数(取决于高度从0开始还是从1开始)
    • 从0开始为边的条数,从1开始为节点数
  • 层序遍历实现求二叉树的深度
class Solution {
public:
    int maxDepth(TreeNode* root) {
        queue<TreeNode*> que;
        if (root) que.push(root);

        int res = 0;
        while (!que.empty()) {
            int size = que.size();
            for (int i = 0; i < size; i++) {
                if (que.front()->left) que.push(que.front()->left);
                if (que.front()->right) que.push(que.front()->right);
                que.pop();
            }
            res++;
        }

        return res;
    }
};
  • 后序遍历实现求二叉树的深度,
    • 实际上是逐步求每个节点的高度,从而最后求出根节点的高度。而根节点的高度正好是二叉树的深度。
class Solution {
public:
    int depthOf(TreeNode* node) {
        if (!node) return 0;
        int leftDepth = depthOf(node->left);
        int rightDepth = depthOf(node->right);
        int depth = 1 + max(leftDepth, rightDepth);
        return depth;
    }
    int maxDepth(TreeNode* root) {
        return depthOf(root);
    }
};

LeetCode559 N叉树的最大深度

559. N 叉树的最大深度 - 力扣(Leetcode)

  • 求树的深度,还是层序遍历更容易扩展到N叉树
class Solution {
public:
    int maxDepth(Node* root) {
        queue<Node*> que;
        if (root) que.push(root);

        int depth = 0;
        while (!que.empty()) {
            int size = que.size();
            depth++;
            for (int i = 0; i < size; i++) {
                for (auto& iter : que.front()->children) {
                    if (iter) que.push(iter);
                }
                que.pop();
            }
        }

        return depth;
    }
};

LeetCode111 二叉树的最小深度

111. 二叉树的最小深度 - 力扣(Leetcode)

  • 二叉树的最小深度是根节点到最近叶节点的经过节点数。

    • 由于我们在递归终止条件中定义空节点的返回值为0(空节点对应一棵空的子树,高度或者深度自然是零)
    • 这会直接影响左右子树的最小深度比较。空节点并不是子树或者叶节点,所以需要我们手动排除空节点,只有左右子树都存在时比较左右子树的深度才有意义。
  • 后序遍历的代码实现

class Solution {
public:
    int minDepthOf(TreeNode* node) {
        if (!node) {
            return 0;
        }
        int leftMinDepth = minDepthOf(node->left);
        int rightMinDepth = minDepthOf(node->right);
        int minDepth = 1;
        if (node->left && !node->right) minDepth += leftMinDepth;
        else if (!node->left && node->right) minDepth += rightMinDepth;
        else minDepth += min(leftMinDepth, rightMinDepth);
        return minDepth;
    }
    int minDepth(TreeNode* root) {
        return minDepthOf(root);
    }
};
  • 层序遍历,在这道题中,层序遍历的代码依然更加直观、更加容易理解。
class Solution {
public:
    int minDepth(TreeNode* root) {
        queue<TreeNode*> que;
        if (root) que.push(root);
        
        int res = 0;
        while (!que.empty()) {
            int size = que.size();
            res++;
            for (int i = 0; i < size; i++) {
                // 到达叶节点则直接返回当前深度
                if (!que.front()->left && !que.front()->right) {
                    return res;
                }
                if (que.front()->left) que.push(que.front()->left);
                if (que.front()->right) que.push(que.front()->right);
                que.pop();
            }
            
        }

        return res;
    }
};

LeetCode222 完全二叉树的节点个数

222. 完全二叉树的节点个数 - 力扣(Leetcode)

  • 关键概念

    • 完美二叉树(Perfect Binary Tree),又称满二叉树(Full Binary Tree):每层的节点数都达到最大值的二叉树就是完美二叉树。
      • 这里我再查阅了一下国外的教材,Full Binary Tree还有完满二叉树的意思,完满二叉树指的是如果一个节点有子节点,则该节点一定既有左孩子又有右孩子。和题目无关,只是定义上的歧义,这里先不讨论。
    • 完全二叉树(Complete Binary Tree),可以由完美二叉树定义,将完美二叉树按层序遍历的顺序,从最后一个节点向前逐个删除节点,得到的二叉树都是完全二叉树。
    • 完全二叉树(Complete Binary Tree):在完全二叉树中,除了最底层节点可能没填满外,其余每层节点数都达到最大值,并且最下面一层的节点都集中在该层最左边的若干位置。若最底层为第 h 层,则该层包含 1~ 2h 个节点。
  • 利用完全二叉树的性质,可以减少计算完全二叉树的节点数需要访问的节点数。

    • 虽然完全二叉树的节点数和层数之间的关系是不确定的,但完全二叉树的子树可能包含完美二叉树。
    • 完美二叉树的节点个数和其层数是确定的,一个完美二叉树的层数为k,则该完美二叉树拥有的节点数为2^k - 1
  • 所以,虽然完全二叉树不一定是完美二叉树,但可以继续判断它的子树是否完美二叉树,如果它的子树是完美二叉树,就可以通过该子树的深度算出子树的节点数。这样就能达到减少访问的节点数的目的。

    • 递归函数的参数和返回值:判断当前子树是否为完美二叉树,如果为完美二叉树,则根据其深度计算节点个数,如果不为完美二叉树,则继续判断其左子树和右子树是否为完美二叉树。参数应为(子)树的根节点,返回值为节点数。
    • 递归的终止条件:一是当前子树的根节点为nullptr,空树返回值为0;而是当前子树为完美二叉树,计算出节点数后直接返回。
    • 单层递归的逻辑:从当前节点出发,计算当前树的左子树的深度和右子树的深度。若左子树的深度和右子树相等,则当前树为完美二叉树,可以直接根据深度计算出节点个数;若当前树不为完美二叉树,则递归判断其左子树和右子树是否为完美二叉树。

递归实现

class Solution {
public:
    int completeTreeNodes(TreeNode* node) {
        if (!node) return 0;
        TreeNode* left = node;
        int leftDepth = 0;
        while (left) {
            left = left->left;
            leftDepth++;
        }
        TreeNode* right = node;
        int rightDepth = 0;
        while (right) {
            right = right->right;
            rightDepth++;
        }
        if (leftDepth == rightDepth) {
            return pow(2, leftDepth) - 1;
        }
        else {
            return completeTreeNodes(node->left) + completeTreeNodes(node->right) + 1;
        }
    }
    int countNodes(TreeNode* root) {
        return completeTreeNodes(root);
    }
};

普通的后序遍历

int postorderCountNodes(TreeNode* node) {
        if (!node) return 0;
        return postorderCountNodes(node->left) + postorderCountNodes(node->right) + 1; 
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容