首页 > 其他分享 >数据结构学习----树的遍历

数据结构学习----树的遍历

时间:2022-09-27 19:49:52浏览次数:54  
标签:node 遍历 TreeNode 递归 res ---- 数据结构 root

树的遍历

前言

  • 在一个平常的星期二下午,一节数据结构课中,想着做点什么的我,打开了力扣。正好老师在讲树,我也从二叉树最基础的遍历开始刷题,没想到打开了新世界的大门······

前提知识

  • 二叉树有三种遍历方式:
    1. 前序遍历(根节点 -> 左子树 -> 右子树)
    2. 中序遍历(左子树 -> 根节点 -> 右子树)
    3. 后序遍历(左子树 -> 右子树 -> 根节点)
  • 可以看出这三种遍历方式的特点:
    1. 前/中/后,代表着根节点的遍历顺序
    2. 左子树一定比右子树先访问到

遍历方法一 ———— 递归

  • 用当时老师的话来说就是:三行代码的事
  • 至于哪三行,话不多说,上代码:
//LeetCode 144. 二叉树的前序遍历
/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode() : val(0), left(nullptr), right(nullptr) {}
 *     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
 *     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
 * };
 */
class Solution {
public:
    void preorder(TreeNode* root, vector<int> &res) {
        //递归退出条件,同时也是“商业程序员”以后工作要注意的错误排查
        if (root == nullptr) {
            return ;
        }
        res.push_back(root -> val);//输出当前节点的值,放第一行表示输出根节点
        preorder(root -> left, res);//遍历左子树
        preorder(root -> right, res);//遍历右子树
    }
    vector<int> preorderTraversal(TreeNode* root) {
        vector<int>res;
        preorder(root, res);
        return res;
    }
};
//时间复杂度:O(n)O(n),其中 nn 是二叉树的节点数。每一个节点恰好被遍历一次。
//空间复杂度:O(n)O(n),为递归过程中栈的开销,平均情况下为 O(\log n)O(logn),最坏情况下树呈现链状,为 O(n)O(n)。
  • 同理,若想使用中/后序遍历,只需将res.push_back(root -> val);放在第二/第三行即可

遍历方法二 ———— 迭代

  • 由之前的刷题经验可知,很多能用递归实现的方法也可以用迭代实现,在此处也是可以的,只不过迭代会比递归更难理解一些

深入底层

  • 为什么能用递归实现的时候,也能用迭代实现呢?我们不妨想想递归实现的底层原理是什么。递归实现遍历本质上是函数不断调用自身,但通过改变传入形参来不断递进的过程。在这个过程中,递归隐式地维护了一个栈:调用preorder并让其入栈,左子树走到底之后就将元素弹出(输出元素),然后遍历右子树。由于这个过程太隐蔽了(都是程序运行时内部完成的),我们就看不到递归背后的运行情况是怎么样的(这可能也是递归难以理解的原因之一)。而迭代其实就是将这个过程展现出来,显式地维护这个栈,完成计算机底层实现的操作,以下是代码:
class Solution {
public:
    vector<int> preorderTraversal(TreeNode* root) {
        vector<int> res;
        if (root == nullptr) {
            return res;
        }

        stack<TreeNode*> stk;
        TreeNode* node = root;
        while (!stk.empty() || node != nullptr) {
            while (node != nullptr) {
                res.emplace_back(node->val);//前序遍历,先将根节点输出
                stk.emplace(node);
                node = node->left;
            }
            node = stk.top();
            stk.pop();
            node = node->right;
        }
        return res;
    }
};
  • 一般都是递归比迭代难理解,但这里似乎是个反例,迭代的代码比递归的更复杂难懂一些,但仔细看会发现和递归其实是一样的,中/后序遍历也只需要简单调整代码逻辑顺序就可以了。

分隔栏:偷个懒下次继续写,该去学java了(2022.9.27 19:28)

遍历方法三 ———— Morris 遍历

遍历方法四 ———— 标记遍历法

标签:node,遍历,TreeNode,递归,res,----,数据结构,root
From: https://www.cnblogs.com/lys-wuyu/p/16735685.html

相关文章

  • 数据扩展
    publicclassDemo02{publicstaticvoidmain(String[]args){//整数扩展进制二进制0b十进制八进制0十六进制0xinti=10;......
  • 第02章 Windows11的基本使用
    第02章Windows11的基本使用第01阶段Java语言核心技术1.Windows11的介绍Windows11是由微软公司(Microsoft)开发的操作系统,应用于计算机和平板电脑等设备。也是全世界......
  • SpringBoot快速入门
    IDEA版本2019.1.3JDK11特别注意:联网!!!引导类SpringBoot项目的入口快速生成一IDEA-->File-->NewModule-->SpringInitializr二NextGroup、Artifact根据项目规......
  • AcWing 算法提高课 线段树+扫描线法 求矩形之并的面积
    例题:求解多个长方形之并的面积https://www.acwing.com/problem/content/249/蓝色表示长方形,红色表示扫描线如下图所示,对于每一个横向的区间,在纵向维护线段树根据纵向......
  • 关于爱情观的一些随笔
           这一段时间忙着写课设,我的电脑重新装机之后,所有的环境变量和ide(集成开发环境)都荡然无存,我花了数天去逃避这件事,然后在一个月黑风高的夜晚,变着花样绕着弯,把所......
  • 流程控制
    流程控制流程控制>>>:控制事物的执行流程 事物执行流程总共可以分为三种 1.顺序结构 从上往下依次执行我们之前所编写的代码都属于该结构 2.分支结构......
  • C#实现定时器的几种方案
    C#实现定时器的几种方案在C#里关于定时器类就有三个1、System.Windows.Forms.Timer2、System.Threading.Timer3、定义在System.Timers.Timer下面对这三个类进行讲解......
  • P2(从2开始的) MAVEN的简介
    注:笔记内容摘抄自视频,侵权删!3项目构建工具摘要:新型公司使用grandle4Maven四大特性4.1依赖管理系统听不太懂,大概就是下载jar包只需要配置,不需要手动下载4.2多模......
  • python垃圾回收机制、流程控制理论、流程控制理论必备知识、分支结构、循环结构
    目录垃圾回收机制1引用计数2标记清除3分代回收流程控制理论流程控制必备知识分支结构循环结构垃圾回收机制一些语言存放空间和释放都需要程序员自己去写代码完成,但是pyth......
  • 实验2:Open vSwitch虚拟交换机实践
    实验2:OpenvSwitch虚拟交换机实践一、实验目的能够对OpenvSwitch进行基本操作;能够通过命令行终端使用OVS命令操作OpenvSwitch交换机,管理流表;能够通过Mininet的Python......