Leetcode 662: 二叉树最大宽度

Leetcode 662: 二叉树最大宽度 是一道考察树的层序遍历和位编号思想的经典题目。其目标是找到二叉树的每一层的最大宽度,宽度定义为:最左节点和最右节点间的节点数(包含这两个节点,中间可以有空节点)


题目描述

  • 给定一个二叉树的根节点 root,返回其最大宽度
  • “宽度”定义为:
    • 每层最左节点(即层级中最左非空节点)到最右节点之间节点的数目。
    • 如果同一层只有一个非空节点,则宽度为 1。

解法 1:层序遍历 + 队列 (广度优先搜索 BFS)

思路
  • 维护位置信息
    • 使用广度优先遍历 BFS,同时为每个节点分配一个基于完全二叉树的编号。根节点的编号为 1,左孩子为 2 * x,右孩子为 2 * x + 1
    • 每层的宽度可通过计算该层最右节点和最左节点对应编号的差值 width = right - left + 1
  • 层序遍历逐层更新最大宽度
    • 使用一个队列,存储每层节点和它们的编号 (node, index)
    • 遍历每一层时,记录最左节点和最右节点对应的编号,然后计算宽度,更新最大值。

代码模板
class Solution {
    public int widthOfBinaryTree(TreeNode root) {
        if (root == null) return 0;

        // 队列存节点和它的"编号" (1-based index for position)
        Deque<Pair<TreeNode, Integer>> queue = new ArrayDeque<>();
        queue.offer(new Pair<>(root, 1));
        int maxWidth = 0;

        while (!queue.isEmpty()) {
            int size = queue.size();
            int left = queue.peekFirst().getValue(); // 当前层最左的编号
            int right = queue.peekLast().getValue(); // 当前层最右的编号
            maxWidth = Math.max(maxWidth, right - left + 1);

            for (int i = 0; i < size; i++) {
                Pair<TreeNode, Integer> pair = queue.poll();
                TreeNode node = pair.getKey();
                int index = pair.getValue();

                // 添加左右孩子到队列
                if (node.left != null) {
                    queue.offer(new Pair<>(node.left, 2 * index));
                }
                if (node.right != null) {
                    queue.offer(new Pair<>(node.right, 2 * index + 1));
                }
            }
        }

        return maxWidth;
    }
}

复杂度分析
  • 时间复杂度:O(n)
    • 每个节点入队和出队各一次,总的时间复杂度为 O(n)。
  • 空间复杂度:O(n)
    • 最差情况下,队列存储所有节点 (如完全二叉树的某一层)。

解法 2:DFS 深度优先搜索 + 位编号

思路
  • 递归方法:
    • 对每个节点,按照广度优先编号规则计算其位置索引,然后递归处理左、右子树。
    • 使用一个 Map<Integer, Integer> 记录每层的最左节点编号。
    • 对每个节点,计算当前宽度:width = currentIndex - leftIndexAtCurrentLevel + 1
    • 同时在递归过程中维护最大宽度值。

关键步骤
  1. 递归传递 当前节点的编号层级信息
  2. 递归更新宽度
    • 如果是当前层第一个访问的节点,记录它的编号为该层的最左编号。
    • 更新当前层宽度,调整全局最大宽度。
  3. 对左右子树分别递归处理:
    • 左子树位置编号为 2 * currentIndex
    • 右子树位置编号为 2 * currentIndex + 1

代码模板
class Solution {
    public int widthOfBinaryTree(TreeNode root) {
        Map<Integer, Integer> leftMostPos = new HashMap<>(); // 每层最左节点的编号
        return dfs(root, 0, 1, leftMostPos);
    }

    private int dfs(TreeNode node, int depth, int index, Map<Integer, Integer> leftMostPos) {
        if (node == null) return 0;

        // 如果当前层是第一次访问,记录最左位置
        leftMostPos.computeIfAbsent(depth, x -> index);

        // 当前宽度 = 当前节点的编号 - 当前层最左编号 + 1
        int currentWidth = index - leftMostPos.get(depth) + 1;

        // 递归求左右子树的最大宽度
        int leftWidth = dfs(node.left, depth + 1, 2 * index, leftMostPos);
        int rightWidth = dfs(node.right, depth + 1, 2 * index + 1, leftMostPos);

        // 返回当前层和左右子树的最大宽度
        return Math.max(currentWidth, Math.max(leftWidth, rightWidth));
    }
}

复杂度分析
  • 时间复杂度:O(n)
    • 每个节点只访问一次。
  • 空间复杂度:O(h)
    • 递归调用栈的最大深度为树的高度,最差情况为 O(n)(如链状树)。

解法 3:BFS 优化(直接计算宽度)

优化点

相比解法 1 的队列实现,这里省略使用 Pair 来记录编号数,直接在每层中计算 startend 的编号即可。


代码模板
class Solution {
    public int widthOfBinaryTree(TreeNode root) {
        if (root == null) return 0;
        Deque<TreeNode> nodeQueue = new ArrayDeque<>();
        Deque<Integer> indexQueue = new ArrayDeque<>();
        nodeQueue.offer(root);
        indexQueue.offer(1);
        int maxWidth = 0;

        while (!nodeQueue.isEmpty()) {
            int size = nodeQueue.size();
            int startIndex = indexQueue.peekFirst();
            int endIndex = indexQueue.peekLast();
            maxWidth = Math.max(maxWidth, endIndex - startIndex + 1);

            for (int i = 0; i < size; i++) {
                TreeNode currNode = nodeQueue.poll();
                int currIndex = indexQueue.poll();

                if (currNode.left != null) {
                    nodeQueue.offer(currNode.left);
                    indexQueue.offer(2 * currIndex);
                }

                if (currNode.right != null) {
                    nodeQueue.offer(currNode.right);
                    indexQueue.offer(2 * currIndex + 1);
                }
            }
        }

        return maxWidth;
    }
}

复杂度分析
  • 时间复杂度:O(n)
    • 每个节点入队、出队各一次。
  • 空间复杂度:O(n)
    • 队列需要存储某一层的所有节点。

快速AC策略

  1. 选择解法:
    • 如果面试中强调代码可读性:写BFS 队列解法(解法 1)。清晰易懂,是优先选择。
    • 如果数据规模中等且需要最优效率:写DFS 解法(解法 2)。递归代码更简洁,但需注意递归深度。
  2. 注意特殊情况
    • 输入为 null 的边界。
    • 确保编号的范围不会超过 int 取值范围(通常树的层级不会超过几十层)。
  3. 模板熟练度
    • BFS 模板:尽量减少额外的数据结构操作(比如 Pair)。
    • DFS 模板:确保层编号和位编号一致处理。

通过以上解法分析与模板准备,可以快速完成这道题并 AC!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于Android平台的SOME/IP测试模块 EPT-ETS

在汽车产业智能化、网联化的时代浪潮中&#xff0c;汽车电子系统正经历着前所未有的变革。SOME/IP&#xff08;Scalable service-Oriented MiddlewarE over IP&#xff09;协议作为汽车电子通信领域的关键技术&#xff0c;其稳定性、可靠性与高效性对于整车性能的提升起着至关重…

【实战 ES】实战 Elasticsearch:快速上手与深度实践-2.2.3案例:电商订单日志每秒10万条写入优化

&#x1f449; 点击关注不迷路 &#x1f449; 点击关注不迷路 &#x1f449; 点击关注不迷路 文章大纲 Elasticsearch批量写入性能调优实战&#xff1a;2.2.3 案例&#xff1a;电商订单日志每秒10万条写入优化1. 原始架构与瓶颈分析1.1 初始集群配置1.2 性能瓶颈定位 2. 全链路…

解决redis lettuce连接池经常出现连接拒绝(Connection refused)问题

一.软件环境 windows10、11系统、springboot2.x、redis 6 7 linux&#xff08;centos&#xff09;系统没有出现这问题&#xff0c;如果你是linux系统碰到的&#xff0c;本文也有一定大参考价值。 根本思路就是&#xff1a;tcp/ip连接的保活(keepalive)。 二.问题描述 在spr…

【开源项目-AI研发】ai-engineer-toolkit

项目地址&#xff08;Fork: 40, Star: 301&#xff09; GitHub - break-into-data/ai-engineer-toolkit: Projects & Resources to help you become a better AI Developer. 项目介绍 官方介绍&#xff1a;帮助你成为更好的 AI 开发者的工具和资源 项目本身是个表格&am…

白帽子讲Web安全资源下载

资源简介 本仓库提供《白帽子讲Web安全》一书的资源下载。这本书由阿里巴巴安全专家刺总编写&#xff0c;是网络安全领域的经典之作&#xff0c;对于从事网络安全工作的专业人士来说是必备的参考资料。 资源描述 书名: 白帽子讲Web安全作者: 阿里巴巴刺总适用人群: 网络安全…

深度学习架构Seq2Seq-添加并理解注意力机制(一)

第一章&#xff1a;人工智能之不同数据类型及其特点梳理 第二章&#xff1a;自然语言处理(NLP)&#xff1a;文本向量化从文字到数字的原理 第三章&#xff1a;循环神经网络RNN&#xff1a;理解 RNN的工作机制与应用场景(附代码) 第四章&#xff1a;循环神经网络RNN、LSTM以及GR…

基于springboot的丢失儿童的基因比对系统(源码+lw+部署文档+讲解),源码可白嫖!

摘要 本丢失儿童的基因比对系统采用B/S架构&#xff0c;数据库是MySQL&#xff0c;网站的搭建与开发采用了先进的Java进行编写&#xff0c;使用了Spring Boot框架。该系统从两个对象&#xff1a;由管理员和用户来对系统进行设计构建。用户主要功能包括&#xff1a;用户注册、登…

Mysql面试篇笔记:

优化&#xff1a; 1.如何定位慢查询&#xff1a; 首先压测接口&#xff0c;查看那个接口比较慢&#xff0c;可以通过多种工具&#xff0c;比如Skywaking 可以查看各个接口响应时间&#xff0c;查看接口最慢&#xff0c;然后去跟踪接口&#xff0c;查看详细信息&#…

嵌入式产品级-超小尺寸游戏机(从0到1 硬件-软件-外壳)

Ultra-small size gaming console。 超小尺寸游戏机-Pico This embedded product is mainly based on miniaturization, followed by his game functions are also very complete, for all kinds of games can be played, and there will be relevant illustrations in the fo…

计算机网络-实验四子网划分

三、实验内容及步骤 1.要求 【题目】某单位申请了⼀个 C 类⽹络&#xff0c;单位内部有3个部门&#xff0c;各部门约50台主机&#xff0c;需要划分为3个⼦⽹&#xff0c;各部门接⼊到汇聚交换机&#xff0c;在汇聚层进⾏路由连通。假定申请到的C类网络为200.200.200.0。 2.实…

deepseek+mermaid【自动生成流程图】

成果&#xff1a; 第一步打开deepseek官网(或百度版&#xff08;更快一点&#xff09;)&#xff1a; 百度AI搜索 - 办公学习一站解决 第二步&#xff0c;生成对应的Mermaid流程图&#xff1a; 丢给deepseek代码&#xff0c;或题目要求 生成mermaid代码 第三步将代码复制到me…

SQL Server2022版+SSMS安装教程(保姆级)

SQL Server2022版SSMS安装教程&#xff08;保姆级&#xff09; 一&#xff0c;安装SQL Server数据库 1.下载安装包 &#xff08;1&#xff09;百度网盘下载安装包 链接&#xff1a;https://pan.baidu.com/s/1A-WRVES4EGv8EVArGNF2QQpwd6uvs 提取码&#xff1a;6uvs &#…

Pany-v2:LFI漏洞探测与敏感文件(私钥窃取/其他)自动探测工具

地址:https://github.com/MartinxMax/pany 关于Pany-v2 Pany-v2 是一款 LFI&#xff08;本地文件包含&#xff09;漏洞探测工具&#xff0c;具备自动识别敏感文件的能力。它能够利用 LFI 漏洞检测并提取 id_rsa 私钥、系统密码文件以及其他可能导致安全风险的敏感信息。该工具…

【音视频】视频基本概念

一、视频的基本概念 1.1 视频码率&#xff08;kb/s&#xff09; 视频码率是指视频文件在单位时间内使用的数据流量&#xff0c;也叫码流率。码率越大&#xff0c;说明单位时间内取样率越大&#xff0c;数据流进度也就越高 1.2 视频帧率&#xff08;fps&#xff09; 视频帧率…

三维数据可视化与表面重建:Marching Cubes算法的原理与应用

1. 引言 随着现代医学影像技术的飞速发展&#xff0c;三维数据的可视化与重建已成为医学研究、临床诊断和手术规划的重要工具。在众多三维重建算法中&#xff0c;Marching Cubes算法因其高效、稳定的特性成为从离散数据场中提取等值面的经典方法。本报告将深入探讨Marching Cu…

探秘基带算法:从原理到5G时代的通信变革【七】FFT/DFT

文章目录 2.6 FFT/DFT2.6.1 离散傅里叶变换&#xff08;DFT&#xff09;2.6.2 快速傅里叶变换&#xff08;FFT&#xff09;2.6.3 方法论与分类体系2.6.4 优缺点与应用2.6.5 实现细节 本博客为系列博客&#xff0c;主要讲解各基带算法的原理与应用&#xff0c;包括&#xff1a;v…

水仙花数(华为OD)

题目描述 所谓水仙花数&#xff0c;是指一个n位的正整数&#xff0c;其各位数字的n次方和等于该数本身。 例如153是水仙花数&#xff0c;153是一个3位数&#xff0c;并且153 13 53 33。 输入描述 第一行输入一个整数n&#xff0c;表示一个n位的正整数。n在3到7之间&#x…

《Python实战进阶》No 7: 一个AI大模型聊天室的构建-基于WebSocket 实时通信开发实战

第7集&#xff1a; 一个AI大模型聊天室的构建-基于WebSocket 实时通信开发实战 在现代 Web 开发中&#xff0c;实时通信已经成为许多应用的核心需求。无论是聊天应用、股票行情推送&#xff0c;还是多人协作工具&#xff0c;WebSocket 都是实现高效实时通信的最佳选择之一。本…

极简Redis速成学习

redis是什么&#xff1f; 是一种以键值对形式存储的数据库&#xff0c;特点是基于内存存储&#xff0c;读写快&#xff0c;性能高&#xff0c;常用于缓存、消息队列等应用情境 redis的五种数据类型是什么&#xff1f; 分别是String、Hash、List、Set和Zset&#xff08;操作命…

ADC采集模块与MCU内置ADC性能对比

2.5V基准电压源&#xff1a; 1. 精度更高&#xff0c;误差更小 ADR03B 具有 0.1% 或更小的初始精度&#xff0c;而 电阻分压方式的误差主要来自电阻的容差&#xff08;通常 1% 或 0.5%&#xff09;。长期稳定性更好&#xff0c;分压电阻容易受到温度、老化的影响&#xff0c;长…