判断二叉树是否为满二叉树
admin
2024-03-15 10:34:40
0

判断二叉树是否为满二叉树

作者:Grey

原文地址:

博客园:判断二叉树是否为满二叉树

CSDN:判断二叉树是否为满二叉树

满二叉树定义

一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是说,如果一个二叉树的层数为K,且结点总数是(2^k) -1 ,则它就是满二叉树。

方法1

使用公式,求二叉树的层数 k, 结点数 n,如果满足(2^k) -1 = n,则为满二叉树。

定义数据结构

    public static class Info1 {public int height;public int nodes;public Info1(int h, int n) {height = h;nodes = n;}}

其中height表示二叉树的层数,nodes表示二叉树的结点个数。

定义递归函数

Info1 process1(Node head)

递归含义是:head 为头的二叉树的层数和点数都是多少,接下来就是 base case

即:head == null的时候,此时,height == 0nodes == 0

        if (head == null) {return new Info1(0, 0);}

接下来是普遍情况

// 去左树上收集信息
Info1 leftInfo = process1(head.left);
// 去右树上收集信息
Info1 rightInfo = process1(head.right);
// 整合成 head 自己的信息
int height = Math.max(leftInfo.height, rightInfo.height) + 1;
int nodes = leftInfo.nodes + rightInfo.nodes + 1;
return new Info1(height, nodes);

方法2

定义如下数据结构

    public static class Info2 {public boolean isFull;public int height;public Info2(boolean f, int h) {isFull = f;height = h;}}

其中isFull表示是否为满二叉树,height表示二叉树的高度。

定义了这个数据结构后,可以梳理可能性,如果以 head 为头的树要符合满二叉树。则需要同时满足下面三种情况

情况1:左树是满二叉树

情况2:右树是满二叉树;

情况3:左右树的高度一样。

定义递归函数

Info2 process2(Node head)

递归含义就是返回以head为头的二叉树Info2结构信息。

base case是

        if (h == null) {return new Info2(true, 0);}

h == null默认是满二叉树,结点个数为0。

接下来是普遍情况,即去左右子树收集相关信息,整合成以h为头二叉树的信息。

    // 去左子树收集相关信息Info2 leftInfo = process2(h.left);// 去右子树收集相关信息Info2 rightInfo = process2(h.right);// 整合成 h 自己的新boolean isFull = leftInfo.isFull && rightInfo.isFull && leftInfo.height == rightInfo.height;int height = Math.max(leftInfo.height, rightInfo.height) + 1;return new Info2(isFull, height);

方法1 和 方法2 的时间复杂度都是O(n),即经过一次后序遍历的时间复杂度。

两种解法的完整代码(含测试代码)如下

public class Code_IsFull {public static class Node {public int value;public Node left;public Node right;public Node(int data) {this.value = data;}}// 第一种方法// 收集整棵树的高度h,和节点数n// 只有满二叉树满足 : 2 ^ h - 1 == npublic static boolean isFull1(Node head) {if (head == null) {return true;}Info1 all = process1(head);return (1 << all.height) - 1 == all.nodes;}public static class Info1 {public int height;public int nodes;public Info1(int h, int n) {height = h;nodes = n;}}public static Info1 process1(Node head) {if (head == null) {return new Info1(0, 0);}Info1 leftInfo = process1(head.left);Info1 rightInfo = process1(head.right);int height = Math.max(leftInfo.height, rightInfo.height) + 1;int nodes = leftInfo.nodes + rightInfo.nodes + 1;return new Info1(height, nodes);}// 第二种方法// 收集子树是否是满二叉树// 收集子树的高度// 左树满 && 右树满 && 左右树高度一样 -> 整棵树是满的public static boolean isFull2(Node head) {if (head == null) {return true;}return process2(head).isFull;}public static class Info2 {public boolean isFull;public int height;public Info2(boolean f, int h) {isFull = f;height = h;}}public static Info2 process2(Node h) {if (h == null) {return new Info2(true, 0);}Info2 leftInfo = process2(h.left);Info2 rightInfo = process2(h.right);boolean isFull = leftInfo.isFull && rightInfo.isFull && leftInfo.height == rightInfo.height;int height = Math.max(leftInfo.height, rightInfo.height) + 1;return new Info2(isFull, height);}// for testpublic static Node generateRandomBST(int maxLevel, int maxValue) {return generate(1, maxLevel, maxValue);}// for testpublic static Node generate(int level, int maxLevel, int maxValue) {if (level > maxLevel || Math.random() < 0.5) {return null;}Node head = new Node((int) (Math.random() * maxValue));head.left = generate(level + 1, maxLevel, maxValue);head.right = generate(level + 1, maxLevel, maxValue);return head;}public static void main(String[] args) {int maxLevel = 5;int maxValue = 100;int testTimes = 1000000;System.out.println("测试开始");for (int i = 0; i < testTimes; i++) {Node head = generateRandomBST(maxLevel, maxValue);if (isFull1(head) != isFull2(head)) {System.out.println("出错了!");}}System.out.println("测试结束");}
}

更多

算法和数据结构笔记

相关内容

热门资讯

linux入门---制作进度条 了解缓冲区 我们首先来看看下面的操作: 我们首先创建了一个文件并在这个文件里面添加了...
C++ 机房预约系统(六):学... 8、 学生模块 8.1 学生子菜单、登录和注销 实现步骤: 在Student.cpp的...
A.机器学习入门算法(三):基... 机器学习算法(三):K近邻(k-nearest neigh...
数字温湿度传感器DHT11模块... 模块实例https://blog.csdn.net/qq_38393591/article/deta...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
Redis 所有支持的数据结构... Redis 是一种开源的基于键值对存储的 NoSQL 数据库,支持多种数据结构。以下是...
win下pytorch安装—c... 安装目录一、cuda安装1.1、cuda版本选择1.2、下载安装二、cudnn安装三、pytorch...
MySQL基础-多表查询 文章目录MySQL基础-多表查询一、案例及引入1、基础概念2、笛卡尔积的理解二、多表查询的分类1、等...
keil调试专题篇 调试的前提是需要连接调试器比如STLINK。 然后点击菜单或者快捷图标均可进入调试模式。 如果前面...
MATLAB | 全网最详细网... 一篇超超超长,超超超全面网络图绘制教程,本篇基本能讲清楚所有绘制要点&#...
IHome主页 - 让你的浏览... 随着互联网的发展,人们越来越离不开浏览器了。每天上班、学习、娱乐,浏览器...
TCP 协议 一、TCP 协议概念 TCP即传输控制协议(Transmission Control ...
营业执照的经营范围有哪些 营业执照的经营范围有哪些 经营范围是指企业可以从事的生产经营与服务项目,是进行公司注册...
C++ 可变体(variant... 一、可变体(variant) 基础用法 Union的问题: 无法知道当前使用的类型是什...
血压计语音芯片,电子医疗设备声... 语音电子血压计是带有语音提示功能的电子血压计,测量前至测量结果全程语音播报࿰...
MySQL OCP888题解0... 文章目录1、原题1.1、英文原题1.2、答案2、题目解析2.1、题干解析2.2、选项解析3、知识点3...
【2023-Pytorch-检... (肆十二想说的一些话)Yolo这个系列我们已经更新了大概一年的时间,现在基本的流程也走走通了,包含数...
实战项目:保险行业用户分类 这里写目录标题1、项目介绍1.1 行业背景1.2 数据介绍2、代码实现导入数据探索数据处理列标签名异...
记录--我在前端干工地(thr... 这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 前段时间接触了Th...
43 openEuler搭建A... 文章目录43 openEuler搭建Apache服务器-配置文件说明和管理模块43.1 配置文件说明...