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

二分法

  • 思路:
    • 明确完全二叉树的定义: 一棵深度为k的有n个结点的二叉树,对树中的结点按从上至下、从左到右的顺序进行编号,如果编号为i(1≤i≤n)的结点与满二叉树中编号为i的结点在二叉树中的位置相同,则这棵二叉树称为完全二叉树。
    • 如果定义根节点深度为 0,则每层第一个节点到最后一个节点的取值为:
      • 2\^h, 2\^(h+1) - 1

    • 可以一直向左遍历出二叉树的深度,则可知这棵树的节点个数范围;
    • 然后通过二分法,确认编号节点是否在二叉树上:
      • 在二叉树上,更新左边界:min = mid;
      • 不在二叉树上,更新右边界:max = mid - 1;
    • 判断编号 k 节点是否在这颗树上: 如果第 k 个节点位于第 h 层,则 k 的二进制表示包含 h+1位,其中最高位是 1,其余各位从高到低表示从根节点到第 k 个节点的路径,0 表示移动到左子节点,1 表示移动到右子节点。通过位运算得到第 k 个节点对应的路径,判断该路径对应的节点是否存在,即可判断第 k 个节点是否存在。
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 (nullptr == root) {
            return 0;
        }

        int depth = 0;
        TreeNode* node = root;
        while (node->left != nullptr) {
            depth++;
            node = node->left;
        }

        // 2^depth
        int min = 1 << depth;
        // 2^(depth + 1) - 1
        int max = (1 << (depth + 1)) - 1;

        while (min < max) {
            int mid = (max - min + 1) / 2 + min;
            if (exist(root, depth, mid)) {
                min = mid;
            } else {
                max = mid - 1;
            }
        }

        return min;
    }

private:
    bool exist(TreeNode* root, int depth, int k) {
        int bits = 1 << (depth - 1);
        TreeNode* node = root;
        while (node != nullptr && bits > 0) {
            if (!(bits & k)) {
                node = node->left;
            } else {
                node = node->right;
            }

            bits >>= 1;
        }

        return (node != nullptr);
    }
};
相关推荐
知识浅谈6 小时前
DeepSeek V4 和 GPT-5.5 在同一天发布了??我也很懵,但对比完我悟了
算法
DeepModel7 小时前
通俗易懂讲透 Q-Learning:从零学会强化学习核心算法
人工智能·学习·算法·机器学习
田梓燊7 小时前
力扣:19.删除链表的倒数第 N 个结点
算法·leetcode·链表
简简单单做算法8 小时前
基于GA遗传优化双BP神经网络的时间序列预测算法matlab仿真
神经网络·算法·matlab·时间序列预测·双bp神经网络
阿豪学编程9 小时前
面试题map/unordered相关
数据结构
guygg889 小时前
利用遗传算法解决列车优化运行问题的MATLAB实现
开发语言·算法·matlab
武藤一雄9 小时前
19个核心算法(C#版)
数据结构·windows·算法·c#·排序算法·.net·.netcore
sali-tec9 小时前
C# 基于OpenCv的视觉工作流-章52-交点查找
图像处理·人工智能·opencv·算法·计算机视觉
梦想的颜色9 小时前
mongoTemplate + Java 增删改查基础介绍
数据结构·数据库·mysql
yu859395810 小时前
MATLAB连续线性化模型预测控制(SL-MPC)
算法·机器学习·matlab