二叉树的层序遍历经典问题(算法村第六关白银挑战)

基本的层序遍历与变换

二叉树的层序遍历

102. 二叉树的层序遍历 - 力扣(LeetCode)

给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]

示例 2:

输入:root = [1]
输出:[[1]]

示例 3:

输入:root = []
输出:[]
public static List<List<Integer>> levelOrder(TreeNode root)
{
    //特判,否则queue.offer(root)会抛出NullPointerException
    if (root == null)
        return new ArrayList<List<Integer>>();

    ArrayList<List<Integer>> ans = new ArrayList<>();

    ArrayDeque<TreeNode> queue = new ArrayDeque<>();
    queue.offer(root);

    while (!queue.isEmpty())
    {
       //获取当前层的结点个数
       int size = queue.size();
            
       //该层结点的值
       ArrayList<Integer> list = new ArrayList<>();

        //遍历当前层
        for (int i = 0; i< size; i++)
        {
            TreeNode t = queue.remove();
            list.add(t.val);

            if (t.left != null)
                queue.offer(t.left);
            if (t.right != null)
                queue.offer(t.right);
        }

        //将这一层结点的值加入答案
        ans.add(list);
    }

    return ans;
}

自底而上的层序遍历

107. 二叉树的层序遍历 II - 力扣(LeetCode)

给你二叉树的根节点 root ,返回其节点值 自底向上的层序遍历 。 (即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历)

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[15,7],[9,20],[3]]

在遍历完一层节点之后,将存储该层节点值的列表 list 添加到结果列表 ans 的头部即可。

为了时间复杂度,ans 使用链式结构的结构。在 ans 头部添加一层节点值的列表 list 的时间复杂度是 O(1)

public List<List<Integer>> levelOrderBottom(TreeNode root)
{
    //特判,否则queue.offer(root)会抛出NullPointerException
    if (root == null)
        return new ArrayList<List<Integer>>();

    ArrayList<List<Integer>> ans = new ArrayList<>();

    ArrayDeque<TreeNode> queue = new ArrayDeque<>();
    queue.offer(root);

    while (!queue.isEmpty())
    {
        //获取当前层的结点个数
        int size = queue.size();

        //该层结点的值
        ArrayList<Integer> list = new ArrayList<>();

        //遍历当前层
        for (int i = 0; i< size; i++)
        {
            TreeNode t = queue.remove();
            list.add(t.val);

            if (t.left != null)
                queue.offer(t.left);
            if (t.right != null)
                queue.offer(t.right);
        }

        //将这一层结点的值插入答案的头部
        ans.add(0,list);
    }

    return ans;
}

锯齿形层序遍历

103. 二叉树的锯齿形层序遍历 - 力扣(LeetCode)

给你二叉树的根节点 root ,返回其节点值的 锯齿形层序遍历 。(即先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行)。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[3],[20,9],[15,7]]
用双端队列维护当前层节点的存储

双端队列可以队头或队尾插入元素。

层序遍历顺序不变,但对当前层节点的存储,我们维护一个变量 isOrderLeft ,记录结点存储是从左至右还是从右至左

  1. 若从左至右,则采用头插法。该层第一个元素在此层遍历结束后,会出现在list的末端
  2. 若从右至左,则采用尾插法。该层第一个元素在此层遍历结束后,会出现在list的首端

最后需要注意的是,往 ans 添加 list 时,需要转换一下 list 的类型

public List<List<Integer>> zigzagLevelOrder(TreeNode root)
    {
        //特判,否则queue.offer(root)会抛出NullPointerException
        if (root == null)
            return new ArrayList<List<Integer>>();

        ArrayList<List<Integer>> ans = new ArrayList<>();

        ArrayDeque<TreeNode> queue = new ArrayDeque<>();
        queue.offer(root);

        boolean isOrderLeft = true;

        while (!queue.isEmpty())
        {
            //获取当前层的结点个数
            int size = queue.size();

            //该层结点的值。用一个双端队列存储
            ArrayDeque<Integer> deque = new ArrayDeque<>();

            //遍历当前层
            for (int i = 0; i< size; i++)
            {
                TreeNode t = queue.remove();

                if (isOrderLeft)
                    deque.offerLast(t.val);
                else
                    deque.offerFirst(t.val);

                if (t.left != null)
                    queue.offer(t.left);
                if (t.right != null)
                    queue.offer(t.right);
            }

            //这一层结点的值经过转换后加入答案
            ans.add(new LinkedList<>(deque));

            isOrderLeft = !isOrderLeft; //交替进行
        }

        return ans;
    }

N叉树的层序遍历

429. N 叉树的层序遍历 - 力扣(LeetCode)

给定一个 N 叉树,返回其节点值的层序遍历。(即从左到右,逐层遍历)。

树的序列化输入是用层序遍历,每组子节点都由 null 值分隔(参见示例)。

示例 1:

img
输入:root = [1,null,3,2,4,null,5,6]
输出:[[1],[3,2,4],[5,6]]

示例 2:

img
输入:root = [1,null,2,3,4,5,null,null,6,7,null,8,null,9,10,null,null,11,null,12,null,13,null,null,14]
输出:[[1],[2,3,4,5],[6,7,8,9,10],[11,12,13],[14]]
结点类型
public class Node
{
    public int val;
    public List<Node> children;

    public Node() {}

    public Node(int _val) {
        val = _val;
    }

    public Node(int _val, List<Node> _children) {
        val = _val;
        children = _children;
    }
}
public List<List<Integer>> levelOrder(Node root)
{
    //特判,否则queue.offer(root)会抛出NullPointerException
    if (root == null)
        return new ArrayList<List<Integer>>();

    ArrayList<List<Integer>> ans = new ArrayList<>();

    ArrayDeque<Node> queue = new ArrayDeque<>();
    queue.offer(root);

    while (!queue.isEmpty())
    {
        //获取当前层的结点个数
        int size = queue.size();

        //该层结点的值
        ArrayList<Integer> list = new ArrayList<>();

        //遍历当前层
        for (int i = 0; i< size; i++)
        {
            Node t = queue.remove();
            list.add(t.val);

            //将当前结点的所有孩子加入队列
            for (Node child : t.children)
            {
                queue.offer(child);
            }
        }

        //将这一层结点的值加入答案
        ans.add(list);
    }

    return ans;
}

处理每层元素

在每个树行中找最大值

515. 在每个树行中找最大值 - 力扣(LeetCode)

给定一棵二叉树的根节点 root ,请找出该二叉树中每一层的最大值。

示例1:

img
输入: root = [1,3,2,5,3,null,9]
输出: [1,3,9]
维护每层的最大值即可
public List<Integer> largestValues(TreeNode root)
{
    //特判,否则queue.offer(root)会抛出NullPointerException
    if (root == null)
        return new ArrayList<>();

    //存储每层结点的最大值
    ArrayList<Integer> list = new ArrayList<>();

    ArrayDeque<TreeNode> queue = new ArrayDeque<>();
    queue.offer(root);

    while (!queue.isEmpty())
    {
        //获取当前层的结点个数
        int size = queue.size();

        //当前层结点的最大值
        int maxOfLevel = Integer.MIN_VALUE;

        //遍历当前层
        for (int i = 0; i< size; i++)
        {
            TreeNode t = queue.remove();

            maxOfLevel = Math.max(maxOfLevel, t.val);

            if (t.left != null)
                queue.offer(t.left);
            if (t.right != null)
                queue.offer(t.right);
        }

        list.add(maxOfLevel);
    }

    return list;
}

二叉树的层平均值

637. 二叉树的层平均值 - 力扣(LeetCode)

给定一个非空二叉树的根节点 root , 以数组的形式返回每一层节点的平均值。与实际答案相差 10-5 以内的答案可以被接受。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[3.00000,14.50000,11.00000]
解释:第 0 层的平均值为 3,1 层的平均值为 14.5,2 层的平均值为 11 。
因此返回 [3, 14.5, 11]
public List<Double> averageOfLevels(TreeNode root)
{
    //特判,否则queue.offer(root)会抛出NullPointerException
    if (root == null)
        return new ArrayList<>();

    //存储每层结点的最大值
    ArrayList<Double> list = new ArrayList<>();

    ArrayDeque<TreeNode> queue = new ArrayDeque<>();
    queue.offer(root);

    while (!queue.isEmpty())
    {
        //获取当前层的结点个数
        int size = queue.size();

        //当前层结点值的和
        double sum = 0;

        //遍历当前层
        for (int i = 0; i< size; i++)
        {
            TreeNode t = queue.remove();

            sum += t.val;

            if (t.left != null)
                queue.offer(t.left);
            if (t.right != null)
                queue.offer(t.right);
        }
		
        //计算平均值并添加到列表
        list.add(sum / size);
    }

    return list;
}

二叉树的右视图

199. 二叉树的右视图 - 力扣(LeetCode)

给定一个二叉树的 根节点 root,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。

示例 1:

img

输入: [1,2,3,null,5,null,4]
输出: [1,3,4]

层序遍历,记录每层最后一个元素即可

public List<Integer> rightSideView(TreeNode root)
{
    //特判,否则queue.offer(root)会抛出NullPointerException
    if (root == null)
        return new ArrayList<>();

    //存储每层的最后一个结点
    ArrayList<Integer> list = new ArrayList<>();

    ArrayDeque<TreeNode> queue = new ArrayDeque<>();
    queue.offer(root);

    while (!queue.isEmpty())
    {
        //获取当前层的结点个数
        int size = queue.size();

        //遍历当前层
        for (int i = 0; i < size; i++)
        {
            TreeNode t = queue.remove();

            //记录当前层最后一个元素
            if (i == size - 1)
                list.add(t.val);

            if (t.left != null)
                queue.offer(t.left);
            if (t.right != null)
                queue.offer(t.right);
        }
    }

    return list;
}

最底层最左边的结点

513. 找树左下角的值 - 力扣(LeetCode)

给定一个二叉树的 根节点 root,请找出该二叉树的 最底层 最左边 节点的值。

假设二叉树中至少有一个节点。

示例 1:

img

输入: root = [2,1,3]
输出: 1

示例 2:

img
输入: [1,2,3,4,null,5,6,null,null,7]
输出: 7
从右往左层序遍历

从右向左层次遍历, 最后一个访问的节点必然是最底层最左侧叶子节点。只需调整一下左右孩子加入队列的次序即可

public int findBottomLeftValue(TreeNode root)
{
    ArrayDeque<TreeNode> queue = new ArrayDeque<>();
    queue.offer(root);

    TreeNode t = null;

    while (!queue.isEmpty())
    {
        //获取当前层的结点个数
        int size = queue.size();

        //遍历当前层
        for (int i = 0; i < size; i++)
        {
            t = queue.remove();

            //右孩子先于左孩子放入队列
            if (t.right != null)
                queue.offer(t.right);
            if (t.left != null)
                queue.offer(t.left);
        }
    }

    return t.val;   //返回最底层最右边的结点的值
}

左叶子之和

404. 左叶子之和 - 力扣(LeetCode)

给定二叉树的根节点 root ,返回所有左叶子之和。

示例 1:

img

输入: root = [3,9,20,null,null,15,7] 
输出: 24 
解释: 在这个二叉树中,有两个左叶子,分别是 915,所以返回 24

提示:

  • 节点数在 [1, 1000] 范围内

广度优先遍历

在BFS的过程中添加一个判断是否为左叶子结点的条件语句即可

public int sumOfLeftLeaves(TreeNode root)
{
    int answer = 0;

    ArrayDeque<TreeNode> queue = new ArrayDeque<>();
    queue.offer(root);


    while (!queue.isEmpty())
    {
        //获取当前层的结点个数
        int size = queue.size();

        //遍历当前层
        for (int i = 0; i < size; i++)
        {
            TreeNode t = queue.remove();

            if (t.left != null)
            {
                //如果t的左孩子是叶子结点
                if(t.left.left == null && t.left.right == null)
                    answer += t.left.val;
                else
                    queue.offer(t.left);
            }
            if (t.right != null)
                queue.offer(t.right);

        }
    }

    return answer;
}

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

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

相关文章

25 心形按钮

效果演示 实现了一个心形的心形图案&#xff0c;当用户点击图案时&#xff0c;图案会旋转并缩小&#xff0c;同时背景颜色会变成白色。 Code <div class"love"><input id"switch" type"checkbox"><label class"love-heart&…

*5.1 Global Memory Bandwidth

并行程序的执行速度可能因计算硬件的资源限制而有很大差异。虽然管理并行代码和硬件资源约束之间的交互对于在几乎所有并行编程模型中实现高性能很重要&#xff0c;但这是一种实用技能&#xff0c;最好通过为高性能设计的并行编程模型中的实践练习来学习。在本章中&#xff0c;…

xss-labs(1-5)

环境准备: 靶场下载:下载仓库 zhangmanhui/xss-labs - Gitee.com 启动phpStudy 搭建将文件解压拉到phpStudy的www目录下就行 最后直接访问:127.0.0.1/xss-labs-master/ 最后再准备一个浏览器的插件用来发送请求:HackBar 插件都配置好了,直接加载到你的浏览器的扩展…

以unity技术开发视角对android权限的讲解

目录 前言 Android权限分类 普通权限 普通权限定义 普通权限有哪些 危险权限 危险权限的定义 危险权限有哪些 动态申请权限实例 申请单个权限实例 第一步&#xff1a;在清单文件中声明权限 第二步&#xff1a;在代码中进行动态申请权限 申请多个权限实例 第一步&am…

【win11 绕过TPM CPU硬件限制安装】

Qt编程指南 VX&#xff1a;hao541022348 ■ 下载iso文件■ 右键文件点击装载出现如下问题■ 绕过TPM CPU硬件限制安装方法■ 虚拟机安装win11 ■ 下载iso文件 选择Windows11 &#xff08;multi-edition ISO&#xff09;在选择中文 ■ 右键文件点击装载出现如下问题 ■ 绕过T…

希尔顿花园酒店喜迎入华十周年里程碑

【2024年1月8日&#xff0c;中国&#xff0c;上海】作为希尔顿集团旗下标志性高端精选服务酒店品牌&#xff0c;希尔顿花园酒店于今年正式迎来其在华经营十周年的里程碑。自2014年中国首家希尔顿花园酒店在深圳开业以来&#xff0c;中国市场已经成为希尔顿花园酒店全球增长的重…

C++中的返回值优化(RVO)

一、命名返回值优化&#xff08;NRVO&#xff09; 是Visual C2005及之后版本支持的优化。 具体来说&#xff0c;就是一个函数的返回值如果是一个对象。那么&#xff0c;正常的返回语句的执行过程是&#xff0c;把这个对象从当前函数的局部作用域&#xff0c;或者叫当前函数的…

2024第15届电子教育、电子商务、电子管理和电子学习国际会议

第十五届电子教育、电子商务、电子管理和电子学习国际会议&#xff08;IC4E 2024&#xff09;将于2024年3月18日-21日在日本福冈举办。本次会议以电子技术为核心&#xff0c;围绕电子教育、电子商务、电子管理以及电子学习等各个方面展开研讨&#xff0c;为相关领域的专家学者们…

【Spring 篇】深入浅出:用Spring注解开发的奇妙之旅

在编程的世界里&#xff0c;Spring框架如同一位慈祥的导师&#xff0c;为我们打开了无尽可能性的大门。而在Spring的广袤领域中&#xff0c;注解是我们最亲密的伙伴之一。本篇博客将深入浅出地介绍使用Spring注解进行开发的奇妙之旅&#xff0c;为你解开注解的神秘面纱。 前奏…

2024年远控软件年度盘点:安全、稳定、功能之选

这目录 前言2024年热门远控软件ToDesk向日葵TeamViewerAnyDeskSplashtopAirDroidChrome Remote DesktopMicrosoft远程桌面RayLinkParallels Access 远程控制软件如何选择&#xff1f;1、功能性2、安全性3、易用性4、稳定性 未来展望与建议结语 前言 随着信息技术不断发展&…

使用命令行方式搭建uni-app + Vue3 + Typescript + Pinia + Vite + Tailwind CSS + uv-ui开发脚手架

使用命令行方式搭建uni-app Vue3 Typescript Pinia Vite Tailwind CSS uv-ui开发脚手架 项目代码以上传至码云&#xff0c;项目地址&#xff1a;https://gitee.com/breezefaith/uniapp-vue3-ts-scaffold 文章目录 使用命令行方式搭建uni-app Vue3 Typescript Pinia V…

移动端对大批量图片加载的优化方法(一)

移动端对大批量图片加载的优化方法&#xff08;一&#xff09;iOS 本篇主要从iOS开发中可以使用到的对大批量图片加载的优化方法进行整理。 1.异步加载 将图片加载任务放在后台线程中进行&#xff0c;避免阻塞主线程&#xff0c;这样可以保证应用的响应性和流畅性&#xff1…

视频剪辑方法:智能转码从视频到图片序列,高效转换攻略

在视频编辑和后期处理中&#xff0c;经常要将视频转换为图片序列&#xff0c;以便进行单独编辑或应用。下面一起来看云炫AI智剪如何批量智能转码的方法&#xff0c;高效地将视频转换为图片序列。 视频转为序列图片缩略图效果 视频转为序列图片的效果图&#xff0c;画面清晰&a…

vue3 img图片怎么渲染

在 Vue3 中加载图片&#xff08;img&#xff09;src地址时&#xff0c;出现无法加载问题。网上很多都建议使用 require 加载相对路径&#xff0c;如下&#xff1a; <img :src"require(../assets/img/icon.jpg)"/>但是按照这种方式加载又会报错如下&#xff1a;…

Access、Trunk、Hybrid接口接收发送数据帧标签剥离区分

以太网二层接口类型 Access Trunk Hybrid 总结&#xff1a; VLAN原理最全最详细讲解&#xff01;彻底搞懂VLAN打和摘tag过程

亚马逊鲲鹏自动测评系统:提升店铺流量与销售的利器

在跨境电商领域&#xff0c;提升店铺流量、排名以及销售业绩一直是卖家们关注的焦点。近期&#xff0c;亚马逊鲲鹏自动测评系统的推出备受关注&#xff0c;成为卖家们提升竞争力的得力工具。据真实客户反馈&#xff0c;该系统不仅能够全自动化批量操作&#xff0c;而且内置了防…

.NET学习教程一——.net基础定义+VS常用设置

一、定义 .NET分为.NET平台和.NET框架。 .NET平台&#xff08;厨房&#xff09;.NET FrameWork 框架&#xff08;柴米油盐酱醋茶&#xff09; .NET平台&#xff08;中国移动联通平台&#xff09;.NET FrameWork 框架&#xff08;信号塔&#xff09; .NET平台基于.NET Fra…

记一次JSF异步调用引起的接口可用率降低

前言 本文记录了由于JSF异步调用超时引起的接口可用率降低问题的排查过程&#xff0c;主要介绍了排查思路和JSF异步调用的流程&#xff0c;希望可以帮助大家了解JSF的异步调用原理以及提供一些问题排查思路。本文分析的JSF源码是基于JSF 1,7.5-HOTFIX-T6版本。 起因 问题背景…

含中间直流的三相电力电子变压器PET仿真模型

微❤关注“电气仔推送”获得资料&#xff08;专享优惠&#xff09; 背景&#xff1a; 目前高压电网中应用的绝大多数电力变压器都属于传 统电力变压器&#xff0c;传统变压器的优势在于工艺简单、安全性 较强。但传统变压器本身的弊端也非常突出&#xff0c;占地大、重 量大&…