222. 完全二叉树的节点个数【 力扣(LeetCode) 】

文章目录

零、原题链接

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

一、题目描述

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

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

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

二、测试用例

示例 1:

cpp 复制代码
输入:root = [1,2,3,4,5,6]
输出:6

示例 2:

cpp 复制代码
输入:root = []
输出:0

示例 3:

cpp 复制代码
输入:root = [1]
输出:1

提示:

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

三、解题思路

3.1 递归

  1. 基本思路:
      一般是采用遍历所有节点的方法来计算节点个数,既然进阶说有非 O ( n ) \Omicron(n) O(n)的复杂度,那一般是利用满二叉树的性质;在满二叉树中,左右子树树高一样,则左子树为完全满二叉树,即节点数为 2 h − 1 2^{h}-1 2h−1;不一样,则右子树为完全满二叉树;所以可以利用这个性质,不断递归,计算子树节点数。
  2. 具体思路:
    • 先计算左右子树树高;
    • 相同,则返回 左子树节点数 + 递归计算右子树节点个数;
    • 不同,则返回 递归计算左子树节点个数 + 右子树节点数;

3.2 二分搜索(非递归)

  1. 基本思路:
      根据左右子树树高,不断缩小最后一行最后一个元素所在的位置的范围;
  2. 具体思路:
    • 先确定初始范围为: [ 0 , 2 ( h − 1 ) ] [0,2^{(h-1)}] [0,2(h−1)]
    • 遍历子树:
      • 左右子树树高相同,最后一个元素在右子树,修改范围下限;
      • 不相同,在左子树,修改范围上限;
    • 总节点数为:树高为 h-1 的完全二叉树的节点数 + 范围上限;

四、参考代码

4.1 递归

时间复杂度: O ( h 2 ) \Omicron(h^2) O(h2)【不断计算子树的树高,即: 1 + 2 + ⋯ + h = h 2 + h 2 1+2+\dots+h=\frac{h^2+h}{2} 1+2+⋯+h=2h2+h】

空间复杂度: O ( h 2 ) \Omicron(h^2) O(h2)【计算树高其实可以用循环解决,这样空间复杂度就只剩 O ( h ) \Omicron(h) O(h)】

cpp 复制代码
/**
 * 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)
            return 0;
        int lh = countH(root->left);
        int rh = countH(root->right);

        if (lh == rh)
            return pow(2, lh) + countNodes(root->right);
        else
            return countNodes(root->left) + pow(2, rh);
    }
    int countH(TreeNode* root) {
        if (!root)
            return 0;

        return countH(root->left) + 1;
    }
};

4.2 二分搜索

时间复杂度: O ( h 2 ) \Omicron(h^2) O(h2)【不断计算子树的树高,即: 1 + 2 + ⋯ + h = h 2 + h 2 1+2+\dots+h=\frac{h^2+h}{2} 1+2+⋯+h=2h2+h】

空间复杂度: O ( 1 ) \Omicron(1) O(1)

cpp 复制代码
/**
 * 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)
            return 0;
        TreeNode* p = root;
        int l = 0;
        int r = pow(2, countH(p) - 1) - 1;

        while (p->left) {
            if (countH(p->left) == countH(p->right)) {
                p = p->right;
                l = (l + r) / 2;
            } else {
                p = p->left;
                r = (l + r) / 2;
            }
        }

        return pow(2, countH(root) - 1) + r;
    }
    int countH(TreeNode* root) {
        int h = 0;
        for (TreeNode* p = root; p; p = p->left) {
            h++;
        }
        return h;
    }
};
相关推荐
半盏茶香20 分钟前
扬帆数据结构算法之雅舟航程,漫步C++幽谷——LeetCode刷题之移除链表元素、反转链表、找中间节点、合并有序链表、链表的回文结构
数据结构·c++·算法
哎呦,帅小伙哦28 分钟前
Effective C++ 规则41:了解隐式接口和编译期多态
c++·effective c++
CodeJourney.40 分钟前
小型分布式发电项目优化设计方案
算法
DARLING Zero two♡1 小时前
【初阶数据结构】逆流的回环链桥:双链表
c语言·数据结构·c++·链表·双链表
9毫米的幻想1 小时前
【Linux系统】—— 编译器 gcc/g++ 的使用
linux·运维·服务器·c语言·c++
带多刺的玫瑰1 小时前
Leecode刷题C语言之从栈中取出K个硬币的最大面积和
数据结构·算法·图论
Cando学算法1 小时前
Codeforces Round 1000 (Div. 2)(前三题)
数据结构·c++·算法
薯条不要番茄酱1 小时前
【动态规划】落花人独立,微雨燕双飞 - 8. 01背包问题
算法·动态规划
小林熬夜学编程1 小时前
【Python】第三弹---编程基础进阶:掌握输入输出与运算符的全面指南
开发语言·python·算法
字节高级特工1 小时前
【优选算法】5----有效三角形个数
c++·算法