news 2026/4/15 17:00:53

【码道初阶】【LeetCode 958】判定完全二叉树:警惕 BFS 中的“管中窥豹”陷阱

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【码道初阶】【LeetCode 958】判定完全二叉树:警惕 BFS 中的“管中窥豹”陷阱

【LeetCode 958】判定完全二叉树:警惕 BFS 中的“管中窥豹”陷阱

在二叉树的面试题中,判定完全二叉树(Check Completeness of a Binary Tree)是一道考察层序遍历(BFS)细节处理的经典题目。

很多同学知道这道题要用队列(Queue)做 BFS,但在处理“何时结束”、“如何判定失败”的逻辑上,很容易掉进思维陷阱。

今天我们就通过两段代码的对比(一段 100% 通过,一段 95% 通过),来揭示算法设计中的局部视野 vs 全局视野的差异。

1. 题目核心难点

完全二叉树的定义通俗来说就是:

  1. 前面的层必须是满的。
  2. 最后一层的节点必须紧凑地靠左排列。
  3. 中间不能有空洞

这意味着,如果我们对树进行层序遍历(BFS),将所有节点(包括空节点null)按顺序放入队列。那么,一旦出现了第一个null,后面就绝对不能再出现任何非null的节点。如果出现了,说明树中间断开了,不是完全二叉树。


2. 案例分析:为什么代码 2 只能通过 95%?

让我们先来看看这段“差一点就对了”的代码(Code 2)。它的思路看起来很机智:在遍历过程中,盯着当前节点和下一个节点看。

❌ Code 2(存在逻辑漏洞)

classSolution{publicbooleanisCompleteTree(TreeNoderoot){Queue<TreeNode>queue=newLinkedList<>();queue.offer(root);while(!queue.isEmpty()){TreeNodecur=queue.poll();// 试图通过“当前节点”和“下一个节点(peek)”的关系直接下定论if(cur==null&&queue.peek()!=null)returnfalse;// 漏洞 1if(cur==null&&queue.peek()==null)returntrue;// 漏洞 2 (致命)// 注意:这里其实还会抛出空指针异常,因为当 cur 为 null 时,// 下面的代码会尝试 cur.left,虽然题目可能保证了数据,但在逻辑上是不严谨的。// 假设我们忽略空指针问题,只看逻辑:if(cur!=null){queue.offer(cur.left);queue.offer(cur.right);}}returntrue;}}

致命死穴:管中窥豹(局部视野)

这段代码最大的问题在于这句话:
if(cur == null && queue.peek() == null) return true;

它的逻辑是:如果我是空的,而且我后面排队的那个人也是空的,那肯定结束了,返回true

反例打击
假设树的层序结构是:[Node, null, null, Node]
这显然不是完全二叉树(中间缺了两个)。

当代码执行到第一个null时:

  1. curnull
  2. queue.peek()是第二个null
  3. 代码直接判定return true

只要连着出现两个空节点,代码 2 就会误判为“通过”,完全忽略了队列后面可能还藏着一个非空的节点!queue.peek()只能看一眼下一个,看不到队尾,这就是典型的“局部视野”导致的错误。


3. 优秀解法:全局扫视(Code 1)

来看看那段完美通过的代码(Code 1)。它的思路非常稳健:分为两个阶段

✅ Code 1(标准答案)

classSolution{publicbooleanisCompleteTree(TreeNoderoot){Queue<TreeNode>queue=newLinkedList<>();queue.offer(root);// 阶段一:无脑入队,直到遇到第一个 nullwhile(!queue.isEmpty()){TreeNodecur=queue.poll();if(cur!=null){// 关键点:不管孩子是不是 null,都放进去!// 这保证了队列里完整保留了树的结构(包括空位)queue.offer(cur.left);queue.offer(cur.right);}else{// 遇到了第一个空节点,阶段一结束!break;}}// 阶段二:排查剩余队列// 既然已经遇到了空节点,那么如果这棵树是完全的,// 队列里剩下的所有东西必须全都是 null。// 只要发现任何一个活着的节点,直接判死刑。while(!queue.isEmpty()){if(queue.peek()!=null){returnfalse;}queue.poll();}returntrue;}}

为什么 Code 1 更优?

  1. 逻辑分层清晰

    • 前半场:正常遍历,一旦遇到空节点,立刻停止“生成新节点”的操作。
    • 后半场:专门负责“验尸”。检查剩下的队列是否纯净(全空)。
  2. 全局视野
    它没有在遇到null时急着下结论,而是用第二个while循环把队列彻底检查一遍。这就避免了“连续两个 null 后面藏着一个 Node”的坑。

  3. 数据结构的正确使用
    它利用了队列 FIFO 的特性,把二叉树拉成了一维的线性结构。判定完全二叉树,本质上就是看这个线性结构中,所有的null是否都集中在最后面

    • [Node, Node, Node, null, null]-> ✅ 是
    • [Node, null, Node, null]-> ❌ 否
    • [Node, null, null, Node]-> ❌ 否(Code 2 会挂在这里)

4. 总结

在解决 BFS 相关题目时,我们要格外注意状态的连续性

  • Code 2 的失败在于试图用O(1)O(1)O(1)的视野(peek)去推断O(N)O(N)O(N)的结局,犯了“急于求成”的错误。
  • Code 1 的成功在于它捕捉到了完全二叉树的本质特征:“遇到空节点后,后续必须全为空”,并用两个循环严谨地实现了这个逻辑。

写算法题时,不要只盯着眼前的peek(),要想到队列深处可能还藏着“魔鬼”

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/14 6:31:04

好消息DataGrip现在对非商业用途免费了,终于可以不用收费的Navicat了

这段时间在整理开发环境的时候&#xff0c;注意到一个消息&#xff1a;DataGrip 已经支持非商业用途免费使用。对经常和数据库打交道的人来说&#xff0c;这个变化还是挺实在的。之前很多人用 Navicat&#xff0c;是因为顺手&#xff0c;但收费一直是绕不开的问题。现在多了一个…

作者头像 李华
网站建设 2026/4/12 17:05:31

ApexCharts.js数据验证终极指南:新手快速解决图表渲染问题

ApexCharts.js数据验证终极指南&#xff1a;新手快速解决图表渲染问题 【免费下载链接】apexcharts.js &#x1f4ca; Interactive JavaScript Charts built on SVG 项目地址: https://gitcode.com/gh_mirrors/ap/apexcharts.js 当你第一次使用ApexCharts.js创建数据可视…

作者头像 李华
网站建设 2026/4/12 14:36:15

终极人体姿态搜索工具:快速实现动作识别与分析的完整指南

终极人体姿态搜索工具&#xff1a;快速实现动作识别与分析的完整指南 【免费下载链接】pose-search x6ud.github.io/pose-search 项目地址: https://gitcode.com/gh_mirrors/po/pose-search pose-search是一款基于现代Web技术的开源人体姿态识别工具&#xff0c;能够实时…

作者头像 李华
网站建设 2026/4/15 14:08:00

通义千问3-VL-Plus - 界面交互

目录 一、概论 二、代码实现 分层设计 模块 1&#xff1a;Request 请求参数封装&#xff08;OparetionRequest&#xff09; 1. 模块定位 2. 核心设计解析 模块 2&#xff1a;Controller 接口层&#xff08;OperationController&#xff09; 1. 模块定位 2. 核心设计解析…

作者头像 李华
网站建设 2026/4/12 11:16:50

终极RefluxJS完全指南:从零开始掌握React数据流管理

RefluxJS是一个专为React应用设计的简单而强大的单向数据流架构库&#xff0c;它让数据管理变得直观易懂。无论你是React新手还是经验丰富的开发者&#xff0c;这份完整指南都将帮助你快速掌握RefluxJS的核心概念和实践技巧。 【免费下载链接】refluxjs A simple library for u…

作者头像 李华
网站建设 2026/4/10 0:20:05

基于51单片机的电子密码锁设计

收藏和点赞&#xff0c;您的关注是我创作的动力 文章目录概要一、系统方案设计2.1系统整体架构设计2.2主控制器方案2.3显示方案设计2.4无线方案设计二、系统电路设计1 锁控制电路设计2 红外遥控接收电路3 系统电路4 系统仿真4.1.1仿真界面说明4.1.2密码输入仿真4.1.3开锁控制仿…

作者头像 李华