【LeetCode】升级打怪之路 Day 14:二叉树的遍历

news2025/1/23 5:00:55

今日题目:

  • 144. 二叉树的前序遍历
  • 94. 二叉树的中序遍历
  • 145. 二叉树的后序遍历
  • 102. 二叉树的层序遍历
  • 107. 二叉树的层序遍历 II
  • 199. 二叉树的右视图
  • 637. 二叉树的层平均值
  • 429. N 叉树的层序遍历
  • 515. 在每个树行中找最大值
  • 116. 填充每个节点的下一个右侧节点指针
  • 117. 填充每个节点的下一个右侧节点指针 II
  • 104. 二叉树的最大深度
  • 111. 二叉树的最小深度

目录

    • Problem 1:二叉树的递归遍历 【easy】
    • Problem 2:二叉树的迭代遍历 【classic】
      • 2.1 前序遍历 迭代版
      • 2.2 中序遍历 迭代版
      • 2.3 后序遍历 迭代版 【必背】
    • Problem 3:二叉树的层次遍历 【classic】
      • LC 102. 二叉树的层序遍历
      • 其他例题

今天主要学习了二叉树的递归遍历、迭代遍历和层序遍历,其中递归遍历和层序遍历都很简单,而迭代遍历的代码写起来稍有困难,这部分需要在理解的基础上,把伪代码背过

Problem 1:二叉树的递归遍历 【easy】

递归遍历二叉树很简单了,可以拿这三个遍历题练练手:

  • 144. 二叉树的前序遍历
  • 94. 二叉树的中序遍历
  • 145. 二叉树的后序遍历

Problem 2:二叉树的迭代遍历 【classic】

在这里插入图片描述

△ 第一次访问; ○ 第二次访问;☆ 第三次访问

2.1 前序遍历 迭代版

144. 二叉树的前序遍历

伪代码思路

void preOrder2(TreeNode T) {
    Stack S;
    TreeNode p = T;

    while (p !=null && !S.empty()) {
        if (p) {
            visit(p);       // 第一次经过时访问之
            S.push(p);      
            p = p.left();   // 一路向左
        } else {
            S.pop(p);
            p = p.right();  // 向右走(step 10)
        }
    }
}

Java 代码实现:

class Solution {

    public List<Integer> preorderTraversal(TreeNode root) {
        if (root == null) {
            return Collections.emptyList();
        }
        
        List<TreeNode> stack = new ArrayList<>();
        List<Integer> result = new ArrayList<>();

        TreeNode p = root;

        while (p != null || !stack.isEmpty()) {
            if (p != null) {
                result.add(p.val);
                stack.addLast(p);
                p = p.left;
            } else {
                p = stack.removeLast();
                p = p.right;
            }
        }

        return result;
    }
}

2.2 中序遍历 迭代版

94. 二叉树的中序遍历

伪代码如下

void inOrder2(TreeNode T) {
    Stack S;
    TreeNode p = T;  // p 是遍历指针
    
    while (p != null || !S.empty()) {  // 栈不空或者 p 不空时循环
        // 一路向左直到空节点
       if (p) {
           S.push(p);       // 当前节点入栈
           p = p.left;      // 向左走
       }
       // 遇到空节点
       else {
           S.pop(p);        // 访问栈顶元素(step9),由于接下来要访问之,故 pop
           visit(p);        // 访问之
           p = p.right;     // 向右子树走(step10)
       }
    }
}

2.3 后序遍历 迭代版 【必背】

145. 二叉树的后序遍历

这个建议直接背过,掌握这个算法思路后,并不难背,大不了多写几遍代码。

算法思路:① 一路向左走并入栈,直到空节点;② 碰到空节点后,读取栈顶元素但不弹出(step9):如果存在右孩子并且未访问过(为了确定之前是从左孩子返回过来的),则向右走;否则,栈顶元素出栈并访问之。

  • 为了区分返回到一个节点时是从左子树回来的还是从右子树回来的,代码设定了辅助指针 recent,它指向最近访问过的节点,当 p.right != recent 时,表示这是从左子树回来的,还没有访问过右子树。

后序遍历迭代版特点

  • 当一个节点的左右子树都被访问后才能出栈(pop)。
  • 实际上,当访问一个节点 p 时,栈中节点恰好是 p 节点的所有祖先,从栈底到栈顶再加上 p 节点,刚好构成从根节点到 p 节点的一条路径。很多算法设计都利用了这一思想,比如求根到某节点的路径,求两个节点的最近公共祖先等。

伪代码如下

void postOrder2(TreeNode T) {
    Stack S;
    TreeNode p = T, recent = null;
    while (p != null && !S.empty()) {
        if (p) {
            S.push(p);
            p = p.left;
        } else {                // 向右
            p = S.top();        // 读取栈顶节点
            if (p.right && p.right != recent) { // 若存在右孩子,且未被访问过
                p = p.right;    // 向右走
            } else {            // 否则弹出节点并访问之
                S.pop(p);
                visit(p);
                recent = p;     // 更新最近访问的节点
                p = null;       // 节点访问完后,重置 p 指针
            }
        } // end else
    } // end while
}

代码实现:

class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<TreeNode> stack = new ArrayList<>();
        List<Integer> result = new ArrayList<>();

        TreeNode p = root, recent = null;

        while (p != null || !stack.isEmpty()) {
            if (p != null) {
                stack.addLast(p);
                p = p.left;
            } else {
                p = stack.getLast();
                if (p.right != null && recent != p.right) {
                    p = p.right;
                } else {
                    result.add(p.val);
                    recent = p;
                    stack.removeLast();
                    p = null;
                }
            }
        }

        return result;
    }
}

Problem 3:二叉树的层次遍历 【classic】

层序遍历的模板可以解决一大类问题,需要谨记。

层次遍历
算法思想

  1. 初始化一个辅助队列 Q;
  2. 根节点入队;
  3. 若 Q 非空,则队头节点出队并访问之,并将其左右孩子入队(如果有的话);
  4. 重复 3 直至队空。

伪代码实现

void levelOrder(TreeNode T) {
    Queue Q;        // 1. 初始化一个辅助队列
    TreeNode p;
    Q.offer(T);      // 2. 根节点入队
    
    while (!Q.empty()) {    // 3. 若 Q 非空,则
        int sz = Q.size();  // 这一层的节点个数
        // 依次将这一层的节点出队
        for (int i = 0; i < sz; i++) {
        	var curr = Q.poll();
        	visit(curr);   // 访问之
        	// 将左右子节点加入队列
        	if (curr.left != null) {
        		Q.offer(curr.left);
        	}
        	if (curr.right != null) {
        		Q.offer(curr.right);
        	}
        }
    }  // 4. 重复直至队空
	
	return;
}

LC 102. 二叉树的层序遍历

102. 二叉树的层序遍历

这是经典使用层序遍历来获取二叉树的层序遍历顺序,基本与模板一致:

class Solution {
    public List<List<Integer>> levelOrder(TreeNode root) {
        if (root == null) {
            return Collections.emptyList();
        }

        Deque<TreeNode> queue = new LinkedList<>();  // 队列
        queue.addLast(root);
        List<List<Integer>> result = new ArrayList<>();

        while (!queue.isEmpty()) {
            int sz = queue.size();
            List<Integer> levelNums = new ArrayList<>();
            for (int i = 0; i < sz; i++) {
                var node = queue.removeFirst();
                levelNums.add(node.val);
                // 将左右子节点加入队列
                if (node.left != null) {
                    queue.addLast(node.left);
                }
                if (node.right != null) {
                    queue.addLast(node.right);
                }
            }
            result.add(levelNums);
        }

        return result;
    }
}

其他例题

借助二叉树的层序遍历的模板,可以一口气解决下面十个题目:

  • 102. 二叉树的层序遍历
  • 107. 二叉树的层序遍历 II
  • 199. 二叉树的右视图
  • 637. 二叉树的层平均值
  • 429. N 叉树的层序遍历
  • 515. 在每个树行中找最大值
  • 116. 填充每个节点的下一个右侧节点指针
  • 117. 填充每个节点的下一个右侧节点指针 II
  • 104. 二叉树的最大深度
  • 111. 二叉树的最小深度

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Fiddler入门:下载、安装、配置、抓包、customize rules

一、fiddler下载安装 安装包下载链接&#xff1a;https://www.telerik.com/download/fiddler 随便选个用途&#xff0c;填写邮箱&#xff0c;地区选择China&#xff0c;勾选“I accept the Fiddler End User License Agreement”&#xff0c;点击“DownLoad for windows”&…

⭐每天一道leetcode:28.找出字符串中第一个匹配项的下标(简单;暴力解;KMP算法,有难度)

⭐今日份题目 给你两个字符串 haystack 和 needle &#xff0c;请你在 haystack 字符串中找出 needle 字符串的第一个匹配项的下标&#xff08;下标从 0 开始&#xff09;。如果 needle 不是 haystack 的一部分&#xff0c;则返回 -1 。 示例1 输入&#xff1a;haystack &q…

3.6作业

作业要求&#xff1a;数据库操作的增、删、改 程序代码&#xff1a; #include<myhead.h> int main(int argc, const char *argv[]) {//定义数据库句柄指针sqlite3 * ppDb NULL;//打开数据库&#xff0c;如果数据库不存在&#xff0c;则创建数据库//将数据库句柄由参数…

移动开发:图像查看器

一、新建ImageViewer模块&#xff0c;添加p1-p9图片(注意mdpi后缀) 二、相关代码 1.MainActivity.java文件代码 package com.example.imageviewer;import androidx.appcompat.app.AppCompatActivity;import android.os.Bundle; import android.view.MotionEvent; import and…

Jacob使用教程--通过宏来寻找变量名

说明: 这里做个随比,参考资料请见前面的系列文章 问题展示: 对于一个操作,当我们不知道怎么利用jacob写代码时,而且网上也找不到,可以按照如下操作: 比如,我们要删除 word中的文本框 我们根本不知道文本框,这个变量叫什么,在Microsoft文档哪个父目录下面, 可以通过…

【MySQL】事务?隔离级别?锁?详解MySQL并发控制机制

目录 1.先理清一下概念 2.锁 2.1.分类 2.2.表锁 2.3.行锁&#xff08;MVCC&#xff09; 2.4.间隙锁 2.5.行锁变表锁 2.6.强制锁行 1.先理清一下概念 所谓并发控制指的是在对数据库进行并发操作时如何保证数据的一致性和正确性。在数据库中与并发控制相关的概念有如下几…

测试遍历1e5,1e8数组耗时

1e8大概0.38秒&#xff0c;即380ms 1e5耗时1ms左右&#xff1a; 代码使用方式来自&#xff1a;clock - C Reference (cplusplus.com)

MicroPython ADX51x读取ID和ADC值

from machine import Pin, SoftSPI import timedef ID(agreement):#txbuf 需要将16进制转换为10进制rxbuf bytearray(3) # create a bufferspi SoftSPI(baudrate200000, polarity1, phase0, mosiPin(16), misoPin(4), sckPin(2))spi.init(baudrate125000) # set the bau…

【前端系列】vue

这里写目录标题 一、Vue简介1.1 主流前端框架/库简介 二、下载和安装Vue2.1 下载2.2 安装完成后&#xff0c;检查2.3创建全局安装目录和缓存日志目录2.4 为了下载包快速&#xff0c;改源为淘宝镜像2.5 查看npm配置修改是否成功 三、配置环境变量环境变量—用户变量—选中Path—…

(C语言)深入理解指针1基础

指针是C语言中的一个难点&#xff0c;但如果真正理解了指针&#xff0c;其实没有很难&#xff0c;本篇文章介绍了指针的基础知识&#xff0c;后面还会继续更行指针的内容。感谢支持。 目录 1. 内存和地址 1.1 内存 1.2 究竟该如何理解编址 2. 指针变量和地址 2.1 取地址操…

Midjourney是个什么软件?midjourney订阅教程

数字时代&#xff0c;AI的应用正在不断推动各个领域的发展。其中&#xff0c;AI在艺术和设计领域的运用引起了广泛的关注。作为一款爆火的AI绘画软件&#xff0c;Midjourney通过其独特的原理和便捷的使用方法&#xff0c;为创作者提供了一个创作逼真绘画作品的全新平台 1、AI绘…

初阶数据结构:二叉树

目录 1. 树的相关概念1.1 简述&#xff1a;树1.2 树的概念补充 2. 二叉树2.1 二叉树的概念2.2 二叉树的性质2.3 二叉树的存储结构与堆2.3.1 存储结构2.3.2 堆的概念2.3.3 堆的实现2.3.3.1 堆的向上调整法2.3.3.2 堆的向下调整算法2.3.3.3 堆的实现 1. 树的相关概念 1.1 简述&a…

Java多线程——信号量Semaphore是啥

目录 引出信号量Semaphore &#xff1f;Redis冲冲冲——缓存三兄弟&#xff1a;缓存击穿、穿透、雪崩缓存击穿缓存穿透缓存雪崩 总结 引出 Java多线程——信号量Semaphore是啥 信号量Semaphore &#xff1f; Semaphore 通常我们叫它信号量&#xff0c; 可以用来控制同时访问特…

Day36 网络概述、IP划分、网络模型

文章目录 网络发展史局域网和广域网局域网&#xff08;LAN&#xff09;广域网&#xff08;Wan&#xff09; 光猫路由器 IP地址基本概念地址划分特殊地址&#xff08;后续编程使用&#xff09;IP地址转换端口字节序 网络模型网络模型OSI模型&#xff08;了解&#xff09;TCP/IP模…

15:Zookeeper高可用集群|分布式消息队列Kafka|搭建高可用Hadoop集群

Zookeeper高可用集群&#xff5c;分布式消息队列Kafka&#xff5c;搭建高可用Hadoop集群 Zookeeper集群Zookeeper角色与特性Zookeeper角色与选举Zookeeper的高可用Zookeeper可伸缩扩展性原理与设计Zookeeper安装zookeeper集群管理 Kafka概述在node节点上搭建3台kafka 高可用Had…

Vue.js+SpringBoot开发高校学院网站

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 学院院系模块2.2 竞赛报名模块2.3 教育教学模块2.4 招生就业模块2.5 实时信息模块 三、系统设计3.1 用例设计3.2 数据库设计3.2.1 学院院系表3.2.2 竞赛报名表3.2.3 教育教学表3.2.4 招生就业表3.2.5 实时信息表 四、系…

循环队列--C语音实现

目录 一、循环队列的特点&#xff1a; 二、普通的队列的缺点&#xff1a; 三、循环队列实现原理&#xff1a; 四、循环队列是实现步骤&#xff1a; 1.循环队列的头文件&#xff1a; 2.循环队列的源文件&#xff1a; 3.注意点&#xff1a; 一、循环队列的特点&#xff1…

灯塔:CSS笔记(1)

CSS&#xff1a;层叠样式表 所谓层叠 即叠加的意思&#xff0c;表示样式可以一层一层的层叠覆盖 css写在style标签中&#xff0c;style标签一般写在head标签里面&#xff0c;title标签下面 <!DOCTYPE html> <html lang"en"> <head><meta cha…

java 从环境变量中获取参数及值

window直接在这设置&#xff1a; linux在/etc/profile文件里存放&#xff1a; export keyvalue 然后立即生效&#xff1a;source /etc/profile 代码获取值这样获取&#xff1a; System.getenv("key");

C++初阶篇----类与对象中卷

目录 引言1. 构造函数1.1概念1.2 特性 2. 析构函数2.1 概念2.2 特性 3. 拷贝构造函数3.1 概念3.2特征 4. 赋值运算符重载4.1 运算符重载4.2 赋值运算符重载4.3 前置和后置重载 5.日期类的实现6.const成员7.取地址及const取地址操作符重载 引言 当一个类既没有成员变量又没有成…