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

. - 力扣(LeetCode)

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

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

思路1:二叉树性质求解,遍历每一个节点。

        递归

                1、确定递归函数的参数和返回值:参数:根节点,返回值:节点数interesting

int CountNodes(TreeNode* root){}

                2、确定终止条件:如果为空节点的话,就返回0,表示节点数为0。

if (root == NULL) return 0;

                3、确定单层递归逻辑:先求它的左子树的节点数量,再求右子树的节点数量,最后取总和再加一 (加1是因为算上当前中间节点)就是目前节点为根节点的节点数量。

int leftNum = CountNodes(root->left);      // 左
int rightNum = CountNodes(root->right);    // 右
int treeNum = leftNum + rightNum + 1;      // 中
return treeNum;
int CountNodes(TreeNode* root){if (root == NULL) return 0;int leftNum = CountNodes(root->left);      // 左int rightNum = CountNodes(root->right);    // 右int treeNum = leftNum + rightNum + 1;      // 中return treeNum;
}
// 简化版本
class Solution {
public:int countNodes(TreeNode* root) {if (root == NULL) return 0;return 1 + countNodes(root->left) + countNodes(root->right);}
};

迭代:二叉树层序遍历(队列)

class Solution {
public:int countNodes(TreeNode* root) {queue<TreeNode*> que;if(root != nullptr)que.push(root);int res = 0;while(!que.empty()){int size = que.size();for(int i = 0; i < size; i++){TreeNode* node = que.front();que.pop();res++;if(node->left) que.push(node->left);if(node->right) que.push(node->right);}}return res;}
};

思路2:满二叉树性质。左0右1,二进制排列,最后一个节点的二进制数就是总节点数。

已知满二叉树,则书高度由左子树深度h决定,所以深度为h的满二叉树的节点总数= 2^h-1,(等比数列求和 (1-2^h)/(1-2) = 2^h-1)。若最底层为第 h 层,则该层节点二进制序号(2^h ~ 2^(h+1))。

求解:总节点数 == h层最后一个节点

      1)遍历左子树,得到树高h,

      2)二分法确定叶子节点的个数(2^h ~ 2^(h+1))(根节点在第0层)

      3)位运算判断叶子节点是否存在。按照二进制编码,左孩子0,右孩子1(第12个节点,二进制表示1100),则2^h每次左移一位位与第k节点,该位为1则右子树,为0则左子树。若节点为空或者2^h左移为0结束循环,返回节点是否为空

class Solution {
public:bool exits(TreeNode* root, int h, int k ){int bits = 1 << (h-1);   // 1<<(h-1) == 2^hTreeNode* node = root;while(node != nullptr && bits > 0){// 该数二进制位,左子树为0,右子树为1// 按位相与,该位同1为1,2&3 == 010 & 011 == 010 == 2if( !(bits & k))  // 检查当前位是否为1node = node->left; //bits & k == 0,即该位为0elsenode = node->right;bits >>= 1; // 判断下一位}return node != nullptr;}int countNodes(TreeNode* root) {if(root == nullptr) return 0;int h = 0; // 默认根节点为第0层TreeNode* node = root;while(node->left != nullptr){h++;node = node->left;}int l = 1 << h, r = (1 << (h+1)) - 1;while(l < r){int mid = (r - l + 1) / 2 + l; // 求中间数,(r+l)/2会有溢位风险if(exits(root, h, mid)){l = mid;}else{r = mid - 1; // mid位置没有节点,所以-1}}return l;}
};

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

两数组根据数组中每条数据对象中的某个值合并去重

假设有两个数组&#xff1a;arr1 和 arr2&#xff0c;并且每个数组中的元素是对象&#xff0c;你希望根据对象中的某个属性&#xff08;比如 id&#xff09;&#xff0c;判断 arr2 中是否已经存在具有相同 id 值的对象。如果没有重复&#xff0c;就将 arr1 中的该对象新增到 ar…

16008.行为树(五)-自定义数据指针在黑板中的传递

文章目录 1.1 背景1.2 xml文件定义1.3 代码实现1.3 执行结果1.1 背景 自定义数据结构指针,通过黑板的形式,在树的节点中进行指针的传递。 1.2 xml文件定义 xhome@ubuntu:~/opt/groot_pro$ cat unit_t1.xml<?xml version="1.0" encoding="UTF-8"?&…

【MongoDB】MongoDB的核心-索引原理及索引优化、及查询聚合优化实战案例(超详细)

文章目录 一、数据库查询效率问题引出索引需求二、索引的基本原理及作用&#xff08;一&#xff09;索引的创建及数据组织&#xff08;二&#xff09;不同类型的索引&#xff08;三&#xff09;索引的额外属性 三、索引的优化与查询计划分析&#xff08;一&#xff09;通过prof…

企业如何实现无缝数据中心进行大数据迁移呢?

数据中心迁移是企业面临的一个复杂而关键的挑战&#xff0c;涉及到大量的数据移动和系统的重新部署。随着业务的扩展和技术的进步&#xff0c;企业可能需要将数据中心迁移到新的位置或升级到更先进的设备。在进行数据迁移时&#xff0c;必须精心规划和执行&#xff0c;以确保数…

FOFA使用教程之从零到精通

FOFA使用教程之从零到精通 前言一、关于网络资产测绘的概念1、啥是网络空间资产测绘2、啥是互联网资产二、FOFA的简要介绍1、FOFA地址是啥?2、关于FOFA的简要介绍三、FOFA精讲1、运算符规则详解① 关于 = 号的使用说明② 关于 == 号的使用说明③ 关于 && 号的使用说明…

初始JavaEE篇 —— 网络编程(2):了解套接字,从0到1实现回显服务器

找往期文章包括但不限于本期文章中不懂的知识点&#xff1a; 个人主页&#xff1a;我要学编程程(ಥ_ಥ)-CSDN博客 所属专栏&#xff1a;JavaEE 目录 TCP 与 UDP Socket套接字 UDP TCP 网络基础知识 在一篇文章中&#xff0c;我们了解了基础的网络知识&#xff0c;网络的出…

【人工智能】10分钟解读-深入浅出大语言模型(LLM)——从ChatGPT到未来AI的演进

文章目录 一、前言二、GPT模型的发展历程2.1 自然语言处理的局限2.2 机器学习的崛起2.3 深度学习的兴起2.3.1 神经网络的训练2.3.2 神经网络面临的挑战 2.4 Transformer的革命性突破2.4.1 Transformer的核心组成2.4.2 Transformer的优势 2.5 GPT模型的诞生与发展2.5.1 GPT的核心…

Webpack 中无法解析别名路径的原因及解决方案

Webpack 中无法解析别名路径的原因及解决方案 文章目录 Webpack 中无法解析别名路径的原因及解决方案1. 引言2. 理解别名路径&#xff08;Alias&#xff09;2.1 什么是别名路径&#xff1f;2.2 别名路径的优势 3. 如何在Webpack中配置别名路径3.1 基本配置3.2 使用别名路径 4. …

最全最简单理解迭代器

1. 迭代器的基础概念(iterator) 1.1 本质 迭代器能够用来遍历容器的对象,与能够遍历数组的指针类似,是广义指针。 1.2 作用: 能够让迭代器与算法不干扰的相互发展,最后又能无间隙的粘合起来。重载了*,++,==,!=,=运算符。用以操作复杂的数据结构。容器提供迭代…

MTSET可溶于DMSO、DMF、THF等有机溶剂,并在水中有轻微的溶解性,91774-25-3

一、基本信息 中文名称&#xff1a;[2-(三甲基铵)乙基]甲硫基磺酸溴&#xff1b;MTSET巯基反应染料 英文名称&#xff1a;MTSET&#xff1b;[2-(Trimethylammonium)ethyl]methanethiosulfonate Bromide CAS号&#xff1a;91774-25-3 分子式&#xff1a;C6H16BrNO2S2 分子量…

CC1链学习记录

&#x1f338; 前言 上篇文章学习记录了URLDNS链&#xff0c;接下来学习一下Common-Colections利用链。 &#x1f338; 相关介绍 Common-Colections是Apache软件基金会的项目&#xff0c;对Java标准的Collections API提供了很好的补充&#xff0c;在其基础上对常用的数据结构…

Android 配置默认输入法

1.背景 最近有个国内的项目&#xff0c;预制了输入法apk&#xff0c;但是无法调出软键盘。原因是没有配置默认输入法&#xff0c;本文主要记录下如何配置默认输入法。 2.代码设置 设置默认输入法需要配置Settings.Secure.ENABLED_INPUT_METHODS和Settings.Secure.DEFAULT_IN…

【juc】ConcurrentHashMap

目录 1.说明2.基本结构3.线程安全机制3.1 分段锁3.2 CAS操作3.3 volatile关键字 4.扩容机制5.其他特性 1.说明 1.ConcurrentHashMap是Java中的一个线程安全的哈希表实现。 2.ConcurrentHashMap的底层结构主要由数组、链表和红黑树组成。 3.在JDK 1.8及之后的版本中&#xff0c;…

数据湖与数据仓库的区别

数据湖与数据仓库是两种不同的数据存储和管理方式&#xff0c;它们在多个方面存在显著的区别。以下是对数据湖与数据仓库区别的详细阐述&#xff1a; 一、数据存储方式 数据仓库 通常采用预定义的模式和结构来存储数据。数据在存储前通常经过清洗、转换和整合等处理&#xff0…

数据结构PTA

20&#xff1a;C 22&#xff1a;B 27&#xff1a;D 填空 4-2&#xff1a;19 4-4&#xff1a;66 4-5&#xff1a;8 5-x&#xff1a;不加分号 ⬇&#xff1a;top p->next 编程 单链表 每个节点除了存放数据元素外&#xff0c;还要存储指向下一节点的指针…

有哪些机器学习实战?——AI实战指南

机器学习已经从理论走向实际应用&#xff0c;各行业的公司和个人都希望通过机器学习来解决现实问题&#xff0c;提升效率。那么&#xff0c;有哪些值得学习和实践的机器学习项目呢&#xff1f;以下将介绍几类热门的机器学习实战项目&#xff0c;涵盖了推荐系统、图像识别、自然…

go语言中的通道(channel)详解

在 Go 语言中&#xff0c;通道&#xff08;channel&#xff09; 是一种用于在 goroutine&#xff08;协程&#xff09;之间传递数据的管道。通道具有类型安全性&#xff0c;即它只能传递一种指定类型的数据。通道是 Go 并发编程的重要特性&#xff0c;能够让多个 goroutine 之间…

Flutter-Padding组件

1. 说明 在html中常见的布局标签都有padding属性&#xff0c;但是Flutter中很多Widget是没有padding属性。这个时候 我们可以用Padding组件处理容器与子元素之间的间距 2. 属性 padding&#xff1a;padding值, EdgeInsetss设置填充的值 child&#xff1a;子组件 3. …

多叉树笔记

1 多叉树定义 多叉树是一种树形结构&#xff0c;它有一个特定的节点被称为“根”节点&#xff0c;而每个节点&#xff08;除了根节点&#xff09;恰好有一个前驱节点&#xff08;父节点&#xff09;。在有根多叉树中&#xff0c;每个节点可以拥有任意数量的后继节点&#xff0…

框架学习04-Spring 事务

1. 事务的基本概念 定义&#xff1a;事务是一组数据库操作的集合&#xff0c;这些操作要么全部成功执行&#xff0c;要么全部不执行。它是为了保证数据的一致性和完整性。例如&#xff0c;在一个银行转账系统中&#xff0c;从一个账户扣款和向另一个账户收款这两个操作应该作为…