当前位置: 首页 > article >正文

二叉树的遍历之迭代遍历

前言:在学习二叉树的时候我们基本上已经了解过二叉树的三种遍历,对于这三种遍历,我们采用递归的思路,很简单的就能实现,那么如何用迭代的方法去解决问题?

我们首先来看第一个:

前序遍历

144. 二叉树的前序遍历 - 力扣(LeetCode)

前序遍历就是以 根 左子树 右子树的顺序遍历二叉树。呢么如何用得带去解决问题呢?

首先我们知道函数递归就是函数一层层的调用自己,本质上是以一种先进后退的思路,而这与栈的特性一致,因此函数递归,我们可以用调用堆栈来看,就是栈的思路。

现在我们解决问题,既然不用递归,但还是可以用到解决问题的本质思路,就是用栈去解决问题:

例图:

 我们就用这个图做演示:

大致解题思路:由于二叉树遍历一直往下走,无法回头,我们需要提供一个栈,假设我们每次先遍历左子树,每次遍历的时候,将我们需要往回遍历的节点放入栈中(往回遍历的节点就是拥有右子树的系点),在遍历完左节点后,此时回退上一个节点,看他的右子树是否为空。

且我们的整个循环的条件中 栈是否为空 就决定是否继续遍历这一个二叉树。

首先前序遍历的顺序 是根  左子树 右子树 。其次,对于二叉树,解决了整个左子树的遍历。那么整个右子树的遍历,也是与之相同的,也就是我们再解决问题时就只针对一个子树,这里我们就以遍历整个左子树为主,实现对整个树的遍历。且题目要求返回数组,因此我们是遍历插入数据。

由于我们无法知道某个左节点的右节点是否为空,所以开始我们先将所有左节点入栈:

第一步操作:

首先要遍历根,也就是从1开始,不为空,放入数组,并入栈,在看它的左,2不为空,放入数组,并入栈,再看它的左,3,不为空,放入数组并入栈。我们就完成了第一步,但是对于最后一个最左节点(它可能是一个子树的根,他也有可能是子树的左节点)。

第二步操作:

从这里开始,我们就要开始判断是否出栈,此时获取当前栈顶元素,3节点,再出栈。(此时栈不为空)

   若3节点无右节点,此时根已经遍历完了,现在轮到左子树了,而3此时是作为一个子树的左节点,并且是遍历顺序左子树中的第一个节点,之后出栈,看节点2的右子树是否存在。

   若3节点有右节点(右子树),比如我们这里就是节点4,3节点就往右子树走,按照前序遍历顺序,我们该将此节点数据放入数组中,因此以该节点为开始,继续第一步操作,将该右节点(也有可能是右子树),按照先把左节点入栈,入数组,在判断右子树情况来出栈。

第三步操作:

其实我们的前序遍历已经实现,可能有人还觉得还没完成,因为右子树还没处理,实际上当我们回退二叉树,也就是出栈的时候,出到最后一个元素,也就是真正的根节点时,栈不为空,循环还没结束,会继续判断当前节点的右子树。

代码实现:

vector<int> preorderTraversal(TreeNode* root) {
    
    //用来返回遍历结果的数组
    vector<int> v;
    
    //用栈来回退我们二叉树
    stack<TreeNode*> t;
    
    TreeNode*node=root;
    //当节点为空或者栈不为空的时候循环继续
    while(node||!t.empty())
    {
       //第一步将左边的节点一次放入数组并入栈 
        while(node)
        {
            v.push_back(node->val);
            t.push(node);
            node=node->left;
        }
        //从该位置开始判断是否要出栈
        node=t.top();
        t.pop();
        if(node->right==nullptr)
         {
            node=nullptr;//直接为空,继续循环进入到出栈操作
         }else
         {
           node=node->right;
         }      
    }
     return v;
    }

中序遍历

中序遍历 的顺序是 左节点 根 右子树 。与前序的插入思想不太一样,不过还是利用栈来回到上一个节点。前序遍历时,其实是有点巧合,因为根节点与左节点连续,我们是直接将左节点一个个入栈后,直接放入数组,因为顺序是从根节点开始,且根节点下来就是左节点,因此插入顺序与我们的直接将左节点插入的顺序一致。

但对于中序和后序遍历,都是先从左节点开始的,因此,从根开始,我们是先将之后的左节点一个个入栈,这里就不能再放数组了,直到最左节点时,根据我们的的遍历顺序,下来时根节点,再来判断。具体分析如下:

还是以这张图为例:

第一步:

刚开始我们还是直接以根为开始,将它的一个个左节点入栈,此时栈中为3,2,1。

第二步:

从这里开始,就需要判断是否出栈,是否插入数组,因此从这里开始,还是先获取栈顶元素,在出栈,当前位置就是节点3,之后就是判断3节点是否具有右子树。

若没有右子树,那么3就是第一个左节点,放入数组,之后回退到上一个节点2,继续判断。

若有右子树,此时节点3是一个根,不能放入数组,我们继续走到节点4,以4节点为开始,与前序遍历一样,执行第一步的操作进行入栈,之后在判断。

直到回退到根节点,此时进行右子树的遍历。

代码如下:

  vector<int> inorderTraversal(TreeNode* root) {
    vector<int> v;
    stack<TreeNode*> t;
    TreeNode* prev=nullptr;
    while(root||!t.empty())
    {
        while(root)
        {
         if(root)
         {
            t.push(root);
         }
         root=root->left;
        }
        root=t.top();
        t.pop();      

        //为空先插左节点
        v.push_back(root->val);
        if(root->right==nullptr)
        {
          root=nullptr;
        }else
        {
          root=root->right;
        }
            
    }
     return v;
    }

 后序遍历

后序遍历的思路与中序遍历有些许一样,但对于情况的判断更加复杂。

因为后序的遍历顺序为 左子树  右子树 根,育贤婿一样,我们还是从根节点开始,入栈左子树的所有左节点,直到最左节点,还是要进行判断。具体分析如下:

 还是一这张图为例:

第一步:

依然是将左子树的所有的左节点依次入栈, 1入,2入,3入,此时到了节点3。

第二步:

从节点3开始判断是否要回退,因此获取栈顶节点为3,出栈,当前节点为3。

若3没有右节点,此时我们就是左节点,我们就可以将3放入数组,之后回退到节点2。

若3的右节点存在,此时的操作是将该节点再次重新入栈(因为下一个右节点(右子树)插入完才轮到我),之后当前节点走到这个右节点,循环回到第一步,继续入栈。

在这里有重要的一点:

第一点:与之前两种遍历一样,我们需要从当前节点回退到上个节点的方法是将指针置空,之后重新赋值为栈顶元素的节点。

其次除了叶子节点容易判断插入,如果上一次插入的节点,是当前节点的右节点,则就需要放入数组里,即 先插的左, 之后插的右,根节点需要判断,当前节点的右节点是上一次插入的节点就说明是根,插入数组。

源码:

vector<int> postorderTraversal(TreeNode* root) {
        stack<TreeNode*>s;
        vector<int> ret;
        TreeNode*prev=nullptr;
     
        while(root||!s.empty())
        {
            //先找所有的左节点
            while(root)
            { 
                s.push(root);
                root=root->left;
            }
            //倒退判断这些节点的右节点,直到最后根节点,在判断右子树
            root=s.top(); 
            s.pop();
            if(root->right==nullptr ||root->right==prev) 
            {
                ret.push_back(root->val);
                prev=root;
                root=nullptr;

            }else
            {
                //若右子树不为空,继续入栈
                 s.push(root);
                root=root->right;
            }
        }
     return ret;
    }


http://www.kler.cn/news/163149.html

相关文章:

  • 文献计量学方法与应用、主题确定、检索与数据采集、VOSviewer可视化绘图、Citespace可视化绘图、R语言文献计量学绘图分析
  • Python嗅探和解析网络数据包
  • 线性回归模型标准公式
  • 解决MySQL字段名与关键字冲突
  • 身份统一管理创新与优化 ——华为云OneAccess应用身份管理服务的2023年
  • cookie总结
  • 什么是自动化测试?什么情况下使用?
  • 【1day】泛微e-office OA系统xml.php 文件 SORT_ID 参数 SQL 注入漏洞学习
  • 计算机基础知识65
  • Linux文件系统与基础IO
  • 【hugging face】bitsandbytes中8 bit量化的理解
  • 在oracle的scn详细说明
  • Kotlin 中密封类、枚举类与密封接口的对比分析
  • Linux——基本指令(一)
  • Nginx按指定格式记录访问日志
  • 联邦多任务蒸馏助力多接入边缘计算下的个性化服务 | TPDS 2023
  • 【LeetCode】28. 找出字符串中第一个匹配项的下标 【字符串单模匹配:KMP算法】
  • Linux设备分类与设备号
  • Django讲课笔记01:初探Django框架
  • 面试宝典之自我介绍
  • 【嵌入式开发 Linux 常用命令系列 4.2 -- .repo 各个目录介绍】
  • 集简云 x 零售企业丨快速集成有赞商城和微盛企微管家,实现私域运营自动化
  • YOLOv8分割训练及分割半自动标注
  • Android MVVM+coroutine+retrofit+flow+hilt
  • LSTM_预测价格问题_keras_代码实操
  • 喜讯:加速度商城系统全系列产品品牌全新升级为Shopfa
  • Java工程找不到javax.xml.bind.annotation包
  • 【flink番外篇】1、flink的23种常用算子介绍及详细示例(3)-window、distinct、join等
  • STM32 map文件详解
  • Kubernetes(K8s 1.27.x) 快速上手+实践,无废话纯享版