LeetCode222_完全二叉树的结点个数

LeetCode222_完全二叉树的结点个数

  • [标签:#位运算 #树 #二分查找 #二叉树](#位运算 #树 #二分查找 #二叉树)
    • [Ⅰ. 题目](#Ⅰ. 题目)
    • [Ⅱ. 示例](#Ⅱ. 示例)
  • [0. 个人方法](#0. 个人方法)

标签:#位运算 #树 #二分查找 #二叉树

Ⅰ. 题目

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

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

Ⅱ. 示例

· 示例 1:

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

输出:6

· 示例 2:

输入:root = []

输出:0

· 示例 3:

输入:root = [1]

输出:1

0. 个人方法

根据完全二叉树最后一层的结点 都集中在该层最左边的若干位置 这一特点来做文章。

  1. 如果左子树和右子树的高度相同(左、右都是满的),左子树一定是一个 满二叉树,节点个数为 2^h - 1。继续递归判断右子树的左子树和右子树...;
  2. 若不同,则右子树是满的,继续递归判断左子树的左子树和右子树...。
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 getHeight(TreeNode* root) {
        int h = 0;
        while (root) {
            h++;
            root = root->left;
        }
        return h;
    }

    int countNodes(TreeNode* root) {
        if (!root) return 0;

        int leftHeight = getHeight(root->left);
        int rightHeight = getHeight(root->right);

        if (leftHeight == rightHeight)
            return (1 << leftHeight) + countNodes(root->right);
        else
            return (1 << rightHeight) + countNodes(root->left);
    }
};

PS:"1 << leftHeight" 表示将 1 左移 leftHeight 位,即表示 2leftHeight(左子树的结点数为 2leftHeight-1,根结点为 1)

  • 时间复杂度分析
    • getHeight() 时间是 O(log n)(树高);

    • 每一层递归时 getHeight 被调用两次,最多递归 log n 层;

    • 总时间复杂度为 O((log n)^2) ------ 比普通 O(n) 的遍历快很多;

相关推荐
YSRM38 分钟前
Leetcode+Java+图论+岛屿问题
java·算法·leetcode·图论
终焉代码44 分钟前
【C++】map与set底层结构——红黑树
开发语言·数据结构·c++
未知陨落2 小时前
LeetCode:89.分割等和子集
算法·leetcode
Cx330❀2 小时前
《C++ STL:vector类(下)》:攻克 C++ Vector 的迭代器失效陷阱:从源码层面详解原理与解决方案
开发语言·数据结构·c++·经验分享·算法
user_huenquan2 小时前
胡恩全10.3作业
开发语言·c++
charlie1145141912 小时前
Windows 10系统编程——进程专题:枚举我们进程的状态
c++·windows·学习·操作系统·进程
bawangtianzun2 小时前
树的重心与直径 性质
数据结构·c++·学习·算法
鄃鳕4 小时前
C++坑系列,C++ std::atomic 拷贝构造函数问题分析与解决方案
java·javascript·c++
code monkey.4 小时前
【探寻C++之旅】第十六章:unordered系列的认识与模拟实现
数据结构·c++·stl
Cx330❀5 小时前
《C++ STL:vector类(上)》:详解基础使用&&核心接口及经典算法题
开发语言·c++·经验分享·算法