《剑指offer(第二版)》面试题55——判断是否为平衡二叉树
一.题目大意
输入一颗二叉树,判断该二叉树是否为平衡二叉树(AVL树)。
二.题解
《剑指offer》上给出了两种解决方式:
1.第一种是从根节点开始,从上往下遍历每个子节点并计算以子节点为根节点的子树的高度,通过判断左右子树的高度差是否大于1来判断是否为AVL树。其中计算子树的高度,利用了TreeDepth函数(具体可见《剑指offer(第二版)》P272).其中TreeDepth的思想如下:
(1)计算一颗二叉树的高度,可以将该问题转化成求子树的高度+1。如果根节点只有左子树的话,那么该二叉树的高度=左子树的高度+1.
(2)如果根节点只有右子树的话,那么该二叉树的高度=右子树的高度+1.
(3)如果根节点既有左子树又有右子树的话,那么该二叉树的高度等于左子树和右子树中高度的较大值加1.
这很显然利用分治的思想就可以解决。
利用TreeDepth函数判断是否为AVL树的话,很多节点往往会重复遍历多次,所以效率会比较低。
2.第二种《剑指offer》上给出的思想是:利用后序遍历的思想,每次遍历到某个节点时,它的左右子树一定已经遍历完了,所以只需在遍历每个节点的时候记录该节点的深度(某一节点的深度等于该节点到叶节点的长度),就可以一边遍历一边判断每个节点是否平衡。
代码可参考书上P274的带代码,但我觉得这种方法不够形象,所以便有了第3种方法。
3.这种方法是这篇文章强调的重点。我们可以利用TreeDepth函数的思想,同样也是分治的思想:
要判断一棵二叉树是否为AVL树,只需判断它的左子树和右子树是否都为AVL数,如果都是AVL树的话,那么这棵树一定是AVL数;对于左子树和右子树的判断进行同样的操作。这就将父问题转换成了一系列的子问题,一旦子问题中有一个不满足的,就返回结果。
代码如下:
class Solution {
public:
bool IsBalanced_Solution(TreeNode* pRoot) {
if(pRoot == nullptr)//此处需要注意,空树的话也是AVL树
return true;
return GetAVLDepth(pRoot) != -1;//返回-1相当于不是AVL树
}
int GetAVLDepth(TreeNode* pRoot)
{
if(pRoot == nullptr)//空的情况返回0
return 0;
int left = GetAVLDepth(pRoot->left);
if(left == -1)//如果子树不是AVL树的话,立马返回,此处相当于剪枝,这步是必须要加的,否则可能出错
return -1;
int right = GetAVLDepth(pRoot->right);
if(right == -1)
return -1;
return abs(left - right) > 1 ? -1 : (max(right,left) + 1);//只有左右子树都为AVL树的话才会返回这棵树的高度
}
};
这种方法是的思想本质与求二叉树的高度的思想是一致的,只不过在返回时,只有左右子树都为AVL树的话才会返回数的高度,否则返回-1(即非AVL树),所以值得回味与思索。还有一点需要注意的是,当根节点为空时,此时是棵空树,但它也是AVL树。最为重要的一点是,如果遇到左子树或者右子树不是AVL树的话,应立马返回-1,否则,后续处理可能出错。
方法3相比方法2在代码上更容易理解,更加形象些。
《剑指offer(第二版)》面试题55——判断是否为平衡二叉树的更多相关文章
- 经典面试题目——找到第n个丑数(参考《剑指offer(第二版)》面试题49)
一.题目大意 给你一个数n,要求返回第n个丑数.其中,丑数的定义如下: 丑数是指只包含因子2.3和5的数.(数字1也是丑数,不过是个特例)引用<剑指offer>上的话来说,对于一个数M,如 ...
- 《剑指offer(第二版)》——面试题36:二叉搜索树与双向链表
具体的题目大意和参考思路在此处不详述(见<剑指offer>),实质就是在中序遍历的过程中调整指针的指向,关于中序遍历有递归和非递归两种操作,所以此处也用了两种方法. 方法1(递归法): 代 ...
- 《剑指offer(第二版)》面试题60——n个骰子的点数
一.题目描述 把n个骰子仍在地上,所有的骰子朝上的一面的点数之和为s,输入n,打印出s所有可能的值出现的概率. 二.题解 <剑指offer>上给出的两种方法,尤其是代码,晦涩难懂且没有注释 ...
- 《剑指offer(第二版)》面试题64——求1+2+...+n
一.题目描述 求1+2+3+...+n,要求不能使用乘除法.for.while.if.else.switch.case等关键字以及条件判断语句 (即三元运算符,A? B : C) 二.题解 虽然求和问 ...
- 结合《剑指offer(第二版)》面试题51来谈谈归并排序
一.题目大意 给定一个数组A,对于数组A中的两个数字,如果排在前面的一个数字大于(必须大于,等于不算)后面的数字,则这两个数字组成一个逆序对.要求输出数组A中的逆序对的总数.例如,对于数组{7,5,6 ...
- 《剑指offer 第二版》题解
剑指Offer 按题号排序 面试题 3:数组中重复的数字 面试题 4:二维数组中的查找 面试题 5:替换空格 面试题 6:从头到尾打印链表 面试题 7:重建二叉树 面试题 8:二叉树的下一个节点 面试 ...
- 剑指offer第二版-10.斐波那契数列
面试题10:斐波那契数列 题目要求: 求斐波那契数列的第n项的值.f(0)=0, f(1)=1, f(n)=f(n-1)+f(n-2) n>1 思路:使用循环从下往上计算数列. 考点:考察对递归 ...
- 剑指offer第二版-5.替换空格
面试题5:替换空格 题目要求: 实现一个函数,把字符串中的每个空格都替换成“%20”,已知原位置后面有足够的空余位置,要求改替换过程发生在原来的位置上. 思路: 首先遍历字符串求出串中空格的数量,求出 ...
- 剑指offer第二版-3.数组中重复的数
面试题3:数组中重复的数 题目要求: 在一个长度为n的数组中,所有数字的取值范围都在[0,n-1],但不知道有几个数字重复或重复几次,找出其中任意一个重复的数字. 解法比较: /** * Copyri ...
随机推荐
- ZK集群搭建和配置
1. 克隆三台空虚拟机(含有jdk和lrzsz),修改网络ip,并关闭虚拟机的防火墙 临时关闭:service iptables stop 永久关闭:chkconfig iptables off 2. ...
- HDU - 5421:Victor and String (回文树,支持首尾插入新字符)
Sample Input 6 1 a 1 b 2 a 2 c 3 4 8 1 a 2 a 2 a 1 a 3 1 b 3 4 Sample Output 4 5 4 5 11 题意:多组输入,开始字符 ...
- HDU2028:Lowest Common Multiple Plus
Problem Description 求n个数的最小公倍数. Input 输入包含多个测试实例,每个测试实例的开始是一个正整数n,然后是n个正整数. Output 为每组测试数据输出它们的最小公倍数 ...
- 模块化,AMD规范
模块化:代码量比较大,可能会有几个人同时写一个页面,同样写在一个文件里面,可能会有冲突,模块化可以解决代码的冲突(匿名函数调用,自己调用自己,就是立即执行函数) 团队的分工更加的明确 异步的执行: A ...
- 容器(container)
一.容器有哪些 平时我们经常看到各种容器名称:Servlet容器.WEB容器.Java WEB容器.Java EE容器等 二.容器是什么 2-1 容器通常理解就是装东西的,我们这里说技术上的容 ...
- Spring——使用自定义标签
文章内容参考了<Spring源码深度解析>一书.自己照着书中内容做了一遍,不懂的地方以及采坑的地方会在文中记录. 推荐一篇post,关于Spring配置文件的命名空间: https://w ...
- 写一个简单脚本检测mysql主从是否正常
这个脚本是在从服务器上执行的 [root@bogon ~]# cat a.sh #!/bin/bash #下面这行目的在于检测mysql端口是否开启,如果没有开启,那说明mysql服务都没有开启,直接 ...
- jenkins 使用smtp2http 邮件服务,扩展灵活的构建通知功能
smtp2http 是一个很方便的可以将smtp 转换为http 服务的工具,同时也支持扩展的开发,我们可以使用此工具 扩展灵活的ci.cd 生命周期管理,而不是简单的邮件处理 备注: 使用docke ...
- drone 1.0 新的定时任务界面&&构建任务支持重启
drone 1.0 的定时任务是一个不错的功能,早期的版本是必须使用cron 表达式的 最近发布的版本支持通过配置就可以了,很方便,只是目前比较简单的,支持小时. 天.周.月.年的模式 环境准备 do ...
- Linux系统运维故障排查
一.思路 1.处理问题要求 2.一般思路 二.具体问题 1.网络问题 (1)网络不通 (2)网络很慢 2.硬件问题 3.操作系统问题 (1)系统无法正常启动 (2)系统运行慢或死机 4.服务或程序问题 ...