【LeetCode经典题解】平衡二叉树高效判断:从O(n²)到O(n)优化

【LeetCode经典题解】平衡二叉树高效判断:从O(n²)到O(n)优化
在这里插入图片描述
🎁个人主页:User_芊芊君子
🎉欢迎大家点赞👍评论📝收藏⭐文章
🔍系列专栏:Java.数据结构
在这里插入图片描述


在这里插入图片描述


【前言】

在数据结构的知识体系里,平衡二叉树是确保二叉树各类操作高效执行的关键存在。力扣平台上“判断二叉树是否为平衡二叉树”这一经典问题,看似解法直观,实则能通过不同的解题思路,清晰展现出算法效率的天差地别,从最开始直观却低效的递归方式,到经过巧妙优化后的高效递归策略,背后蕴含着对平衡二叉树本质的深度剖析与精准把握。

这里写目录标题

一、平衡二叉树

平衡二叉树又称AVL树:当一棵空树或者它的左右两棵字数的高度差的绝对值不超过一,并且两棵子树都是平衡二叉树
【注意】

任意节点的左右子树高度差不超过一;所以子树都满足平衡条件;
在这里插入图片描述

二、思路分析

方法一:时间复杂度为O(n^2)

首先判断根节点是否为空,跟节点为空的话,是平衡二叉树;然后就是用getHigh()方法获取左右子树的高度;判断他们的高度差的绝对值小于等于1的话,还要递归左右子树,左右子树平衡就是平衡二叉树,;时间复杂度为O(n^2),因为计算高度时会重复遍历子树,每个节点都要求一次高度
在这里插入图片描述

方法二:时间复杂度为O(n)

先判断根节点是否为空,为空就是平衡二叉树;不为空调用getHigh()方法,通过判断getHigh()的返回值是否大于等于0来判断是否平衡;getHigh()方法计算二叉树高度的同时,判断树是否平衡,平衡返回树的高度,不平衡返回-1时间复杂度为O(n),n是二叉树节点,==每个节点只被访问一次,计算高度和判断平衡的操作都是O(1);

三、代码展示

publicbooleanisBalanced(TreeNode root){if(root ==null){returntrue;}int leftH =getHigh(root.left);int rightH =getHigh(root.right);returnMath.abs(leftH - rightH)<=1&&isBalanced(root.left)&&isBalanced(root.right);}publicintgetHigh(TreeNode root){if(root ==null){return0;}int leftH =getHigh(root.left);int rightH =getHigh(root.right);return leftH > rightH ? leftH+1: rightH+1;}
publicbooleanisBalanced(TreeNode root){if(root ==null){returntrue;}returngetHigh(root)>=0;}publicintgetHigh(TreeNode root){if(root ==null){return0;}int leftH =getHigh(root.left);if(leftH <0){return-1;}int rightH =getHigh(root.right);if(leftH >=0&& rightH >=0&&Math.abs(leftH - rightH)<=1){return leftH > rightH ? leftH+1: rightH+1;}else{return-1;}}

四、总结

“自顶向下”的递归方法虽然思路直观易懂,但由于在计算过程中会重复去计算子树的高度,导致时间复杂度达到了O(n^2),在处理大规模数据时显得力不从心;而“自底向上”的递归方法,借助后序遍历的顺序,在计算子树高度的同时就完成了平衡与否的判断,还利用提前终止递归的技巧避免了重复计算,成功将时间复杂度优化到O(n),这一过程很好地体现了在算法设计中,通过调整执行逻辑、复用中间计算结果,能够显著提升算法效率的核心思想。

Read more

一序平衡,括号归真:单括号匹配算法的优雅美学

一序平衡,括号归真:单括号匹配算法的优雅美学

一序平衡,括号归真:单括号匹配算法的优雅美学 * 一、问题本源:何为合法的单括号序列? * 🔹 核心判定准则 * 举个直观例子 * 二、思维演进:从朴素思路到极致优化 * 1. 原始思路:双计数器法(直观但冗余) * 2. 优雅优化:单计数器法(平衡值思想) * 三、图形化拆解:算法执行全过程 * 案例1:合法序列 `( () () )` * 案例2:非法序列 `( ) )` * 四、算法原理深度讲解 * 五、C++ 关键代码实现与讲解 * 代码核心亮点 * 六、算法思维的启示 * 七、结语 算法与数据结构片头 在算法的星河里,单括号匹配问题如一颗温润的贝珠,以极简的形态承载着最纯粹的算法思维。它剥离了复杂的类型干扰,只保留一对小括号 () 的序列判断,却能让我们窥见「平衡思想」与「代码优化」的本质——这是编译器语法校验、表达式合法性判断的底层基石,

By Ne0inhk
C语言指针与数组的深度应用与内存解析

C语言指针与数组的深度应用与内存解析

C语言指针与数组的深度应用与内存解析 💡 学习目标:掌握指针与数组的等价性原理,熟练运用指针操作数组元素,理解二者在内存中的存储本质,解决实际开发中数组遍历、数据拷贝的高效实现问题。 💡 学习重点:指针与数组名的区别、指针算术运算操作数组、二维数组的指针访问方式、内存视角下的数组与指针关系。 48.1 指针与数组的核心关联:本质与等价性 在C语言中,指针和数组的关系密不可分。很多初学者会混淆数组名和指针的概念,实际上二者既有联系又有本质区别。 48.1.1 数组名的“隐式转换”特性 当数组名出现在表达式中时,它会隐式转换为指向数组首元素的指针。我们可以通过一个简单的例子来验证这个特性: #include<stdio.h>intmain(){int arr[5]={10,20,30,40,50};// 输出数组首元素地址printf("数组名arr的地址:%p\n", arr)

By Ne0inhk
数据结构—顺序表超经典算法

数据结构—顺序表超经典算法

数据结构—顺序表链表经常用到的算法 * 所有题目链接 * 顺序表算法题(双指针法) * 移除元素 * 删除有序数组中的重复项 * 合并两个有序数组 * 链表算法题(快慢指针,三指针法,创建新链表法) * 移除链表元素 * 反转链表 * 链表的中间节点 * 合并两个有序链表 * 链表分割 * 链表的回文结构 * 相交链表 * 环形链表(快慢指针) * 环形链表I * 环形链表II * 代码仓库 所有题目链接 移除元素 删除有序数组中的重复项 合并两个有序数组 移除链表元素 反转链表 链表的中间节点 合并两个有序链表 链表分割 链表的回文结构 相交链表 环形链表I 环形链表II 顺序表算法题(双指针法) 移除元素 题目链接↓ 移除元素 题目讲解↓ 思路:双指针法,创建两个变量dst,src如果src指向的数据是val,src++如果src指向的数据不是val,赋值(

By Ne0inhk
【数据结构和算法】面试必刷之随机链表复制:这三步让你彻底吃透 random 指针

【数据结构和算法】面试必刷之随机链表复制:这三步让你彻底吃透 random 指针

🔥小龙报:个人主页 🎬作者简介:C++研发,嵌入式,机器人等方向学习者 ❄️个人专栏:《C语言》《【初阶】数据结构与算法》 ✨ 永远相信美好的事情即将发生 文章目录 * 前言 * 一、随即链表的复制 * 1.1 题目 * 1.2 算法原理 * 1.3 代码 * 总结与每日励志 前言 随机链表的复制是数据结构中的经典难题,核心难点在于复制节点的random指针——其指向的节点可能尚未创建,也可能指向链表中的任意节点。本文采用“原地拷贝+拆分”的最优思路,分三步拆解解题逻辑,结合代码实现与原理分析,清晰讲解如何高效解决该问题,帮助读者吃透random指针的处理技巧,掌握链表操作的核心思维。 一、随即链表的复制 1.1 题目 链接:随机链表的复制 1.2 算法原理

By Ne0inhk