二叉树三种遍历的递归与非递归写法

news2025/1/15 6:45:50

目录

​编辑

一,前序遍历

题目接口:

递归解法:

非递归解法:

二,中序遍历

题目接口:

递归解法:

非递归写法:

三,后序遍历

题目接口:

递归解法:

非递归解法:


一,前序遍历

题目接口:

/**
 * 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:
    vector<int> preorderTraversal(TreeNode* root) {
        

    }
};

递归解法:

对于这道题,相信大家都能够轻松解决掉。递归写法,非常简单:

class Solution {
public:
    void  _preorderTraversal(TreeNode*root, vector<int>&ret)
     {
         if(root == nullptr)
         {
             return ;
         }
          
          ret.push_back(root->val);

           _preorderTraversal(root->left, ret);

           _preorderTraversal(root->right,ret);

     }

    vector<int> preorderTraversal(TreeNode* root) {
         
         vector<int>ret;
         _preorderTraversal(root,ret);

         return ret;

    }
};

非递归解法:

但是,如果要你写出一个非递归版本的的写法呢?我们该如何写呢?步骤如下:

1. 搞一个栈,这个栈的作用是存下每一个节点。

2.定义一个cur指针,指向当前节点。然后从该节点cur开始,使用一个小循环循环遍历左子树,在将一个左子树遍历完以后也就是遍历到nullptr以后便结束循环。

3.取栈顶元素top,让cur重新指向top的右指针。然后从新的cur开始重新遍历左子树。

4.当栈为空且cur为nullptr时便可以结束大循环,返回得到的前序遍历的结果。

代码如下:

class Solution {
public:
    vector<int> preorderTraversal(TreeNode* root) {
       
        vector<int>ret;//存储结果的数组
        stack<TreeNode*>st;//栈
        TreeNode*cur = root;

        while(!st.empty()||cur)//循环结束条件,必须在两者都是nullptr的情况下才能结束循环。
        {
            while(cur)
            {
                ret.push_back(cur->val);
                st.push(cur);
                cur = cur->left;
            }
             
             TreeNode* top = st.top();
             st.pop();
             cur = top->right;//指向右节点,遍历右树。
        }

        return ret;
        

    }
};

总结,这里的关键一步便是遍历每一个节点的左树。然后将每一个节点用栈记录下来。这里为什么要使用栈呢?这是利用了栈后进先出的特点!!!由于在电脑上画图比较麻烦,所以大家可以自己根据这个代码画图模拟一下。

二,中序遍历

题目接口:

/**
 * 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:
    vector<int> inorderTraversal(TreeNode* root) {

    }
};

递归解法:

使用递归解法任仍然是简单的,也就是按照顺序左子树->根->右子树的顺序来递归遍历这棵二叉树。代码如下:

class Solution {
public:
    void _inorderTraversal(TreeNode*root, vector<int>&in)
    {
        if(root == nullptr)
        {
            return;
        }
        
        _inorderTraversal(root->left,in);

        in.push_back(root->val);

        _inorderTraversal(root->right,in);

    }

    vector<int> inorderTraversal(TreeNode* root) {
           vector<int>in;
           _inorderTraversal(root,in);
           return in;
    }
};

非递归写法:

 有了上面的前序遍历的非递归写法的思想以后,中序遍历的非递归写法就好写很多了。我们只需要在前序遍历的非递归写法上改一下根节点插入到in数组中的顺序便可以了。代码如下:

class Solution {
public:
    vector<int> inorderTraversal(TreeNode* root) {
        vector<int>ret;//存储结果的数组
        stack<TreeNode*>st;//栈
        TreeNode*cur = root;

        while(!st.empty()||cur)//循环结束条件,必须在两者都是nullptr的情况下才能结束循环。
        {
            while(cur)//这个循环只往栈st里面插入插入节点的指针而不往ret里面插入值
            {
                st.push(cur);
                cur = cur->left;
            }
             
             TreeNode* top = st.top();
             ret.push_back(top->val);//在这里才插入值
             st.pop();
             cur = top->right;//指向右节点,遍历右树。
        }

        return ret;
        

    }
};

三,后序遍历

题目接口:

/**
 * 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:
    vector<int> postorderTraversal(TreeNode* root) {

    }
};

递归解法:

这道题的递归解法仍然很简单,就是按照左子树->右子树->根的顺序遍历这棵二叉树。递归代码如下:

class Solution {
public:
    void _postorderTraversal(TreeNode*root,vector<int>&ret)
    {
           if(root == nullptr)
           {
               return;
           }

           _postorderTraversal(root->left,ret);
           _postorderTraversal(root->right,ret);
           ret.push_back(root->val);
    }
    
    vector<int> postorderTraversal(TreeNode* root) {
        
           vector<int>ret;
           _postorderTraversal(root,ret);
           return ret;
    }
};

非递归解法:

这道题难就难在非递归解法的代码我们该如何去写呢?难就难在这里了。首先我们也都知道,后序布遍历的遍历顺序是:左子树->右子树->根。所以我们仍然要先访问左子树。我们仍然要先访问左,先把所有的左节点插入到栈里面。这一步其实和前面的中序遍历与前序遍历的思路是一样的,但是在后序遍历里面是否能够访问当前节点是要做判断的:当前节点必须在右节点被访问以后才能访问。

代码如下:

class Solution {
public:
    vector<int> postorderTraversal(TreeNode* root) {

        stack<TreeNode*>st;
         vector<int>ret;
        TreeNode*cur = root;
        TreeNode*prev = nullptr;//记录前面访问了那一个节点

        while(!st.empty()||cur)
        {
            while(cur)//只插入不访问
            {
                st.push(cur);
                cur = cur->left;
            }
            
            TreeNode* top = st.top();//找到最后一个插入栈的节点
            if(prev == top->right)//如果这个节点的右节点已经被访问过来,这个节点便是可以访问的
            {
                prev = top;//更新prev
                ret.push_back(top->val);
                st.pop();
            }

            else//如果这个节点的右节点没有被访问过,便先访问右节点(右树)
            {
                  cur = top->right;
                  prev = cur;//更新prev
            }
        }

        return ret;

    }
};

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

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

相关文章

IOC课程整理-17 Spring事件

1. Java 事件/监听器编程模型 2. 面向接口的事件/监听器设计模式 3. 面向注解的事件/监听器设计模式 4. Spring 标准事件-ApplicationEvent 5. 基于接口的 Spring 事件监听器 6. 基于注解的 Spring 事件监听器 7. 注册 Spring ApplicationListener 8. Spring 事件发布器 9. Spr…

基于VectorGrid加载GeoServer发布的矢量瓦片实例

目录 前言 一、关于VectorGrid 1、开源地址 2、本地示例 二、与LeafLet集成 1、新建html页面 2、地图初始化 3、pbf瓦片地址配置 4、pbf初始化 三、GeoServer跨域问题 1、web.xml配置 2、重启tomcat 总结 前言 回望10月&#xff0c;发生了一些变动&#xff0c;面向未…

状态机图和活动图

在面向对象软件分析过程中&#xff0c;状态机图和活动图用于建立软件的动态模型&#xff0c;主要描述系统随时间变化的行为。 1.状态图 1.1概念 状态图用来描述对象状态和事件之间的关系&#xff0c;强调一个实体基于事件反应的动态行为。状态图适合用于表述在不同用例之间的…

C语言#error和#line

C语言#error和#line #error #error用于生成一个编译错误消息&#xff0c;并停止编译 示例&#xff1a; 随便找了一个工程测试下#error 看图中我圈起来的部分&#xff0c;编译器提示warning和error。看我的程序如果没有定义TEST_#ERROR这个宏&#xff0c;编译器会报错You di…

我是如何快速从python小白达到20k?

前言 首先说一下我自己的情况&#xff0c;我之前是学JAVA的&#xff0c;JAVA亡了只好转行python 很多新手就在好奇自己明明都认认真真的学习了python&#xff0c;但就是感觉很杂很乱&#xff0c;按照我这个流程&#xff0c;至少可以省一大半时间&#xff0c;完整的知识体系很重…

LED主流光源-环形光源

1&#xff09;产品特点&#xff1a; ① 环形光源提供不同角度照射&#xff0c;能突出物体的三维信息有效的解决对角照射阴 影问题&#xff1b; ② 周围表面采用滚花设计&#xff0c;扩大散热面积保障光源的使用寿命&#xff1b; ③ 根据客户不同需求可 选配不同漫射板&#xff…

AIGC如何助力产品研发的创新和性能提升

1、现有的产品和系统的升级 a&#xff09;、关键算法的替换&#xff0c;用深度学习来替换&#xff0c;用数学来描述&#xff1a; 需要定义好中间状态的和&#xff0c;总体过程是,中间的过程,替换为。 总体过程表示成下面的方式: 完成替换过程&#xff1a; 。 b&#xff09;…

vivado窗口使用与分析2-IDE 中的逻辑分析

逻辑分析 包括 &#xff1a; • “ Netlist ”窗口 • “ Hierarchy ”窗口 • “ Schematic ”窗口 1、 “ Netlist ”窗口 “ Netlist ” &#xff08; 网表 &#xff09; 窗口显示了网表中由综合工具所处理的设计层级。 根据综合设置 &#xff0c; 网表层级与原始 RT…

【网络】序列化反序列化

序列化反序列化 一、序列化反序列化1、概念2、序列化作用3、序列化框架的选择 二、Json1、介绍2、简单使用 一、序列化反序列化 1、概念 在前文《网络编程套接字》中&#xff0c;我们实现了服务器与客户端之间的字符串通信&#xff0c;这是非常简单的通信&#xff0c;在实际使…

JavaScript从入门到精通系列第二十五篇:JavaScript中的Date对象

文章目录 一&#xff1a;Date对象简介 1&#xff1a;概念简介 二&#xff1a;Date对象 1&#xff1a;创建当前时间 2&#xff1a;创建指定时间 三&#xff1a;日期对象函数 1&#xff1a;getDate() 2&#xff1a;getDay() 3&#xff1a;getMonth() 4&#xff1a;getF…

基于springboot环保话题管理系统-计算机毕设 附源码 28550

springboot环保话题管理系统 摘 要 随着科学技术的飞速发展&#xff0c;社会的方方面面、各行各业都在努力与现代的先进技术接轨&#xff0c;通过科技手段来提高自身的优势&#xff0c;中小型企业当然也不能排除在外。环保话题管理系统是以实际运用为开发背景&#xff0c;运用软…

Flask路由机制分析之二

一、前言 上篇 《Flask 路由机制分析之一》主要讲了Python函数的特性以及装饰器的基本概念&#xff0c;这节我们具体分析一下路由内部机制&#xff0c;Flask路由依赖于werkzegu的routing模块来实现。 二、werkzegu的routing模块介绍 Werkzegu库的routing模块主要功能在于URL…

iMazing2024年最新许可证-iMazing许可证激活补丁

《iMazing》2024年最新许可证&#xff0c;iMazing 号称是 Mac 和PC上最好的iOS设备管理器。iMazingapp还可以帮助用户轻松地管理应用程序&#xff0c;可以方便的进行拷贝&#xff0c;安装&#xff0c;删除和更新应用程序。它可以帮助用户轻松访问文件系统&#xff0c;并可以轻松…

使用FastAPI部署Ultralytics YOLOv5模型

YOLO是You Only Look Once(你只看一次)的缩写&#xff0c;它具有识别图像中的物体的非凡能力&#xff0c;在日常应用中会经常被使用。所以在本文中&#xff0c;我们将介绍如何使用FastAPI的集成YOLOv5&#xff0c;这样我们可以将YOLOv5做为API对外提供服务。 Python有几个web框…

深入探究Python中的深度学习:神经网络与卷积神经网络

当下&#xff0c;深度学习已经成为人工智能研究和应用领域的关键技术之一。作为一个开源的高级编程语言&#xff0c;Python提供了丰富的工具和库&#xff0c;为深度学习的研究和开发提供了便利。本文将深入探究Python中的深度学习&#xff0c;重点聚焦于神经网络与卷积神经网络…

基于springboot实现校园台球厅人员与设备系统项目【项目源码+论文说明】计算机毕业设计

基于springboot实现校园台球厅人员与设备系统演示 摘要 在Internet高速发展的今天&#xff0c;我们生活的各个领域都涉及到计算机的应用&#xff0c;其中包括校园台球厅人员与设备管理系统的网络应用&#xff0c;在外国管理系统已经是很普遍的方式&#xff0c;不过国内的管理网…

CN考研真题知识点二轮归纳(1)

本轮开始更新真题中涉及过的知识点&#xff0c;总共不到20年的真题&#xff0c;大致会出5-10期&#xff0c;尽可能详细的讲解并罗列不重复的知识点~ 目录 1.三类IP地址网络号的取值范围 2.Socket的内容 3.邮件系统中向服务器获取邮件所用到的协议 4.RIP 5.DNS 6.CSMA/CD…

C++标准模板(STL)- 类型支持 (类型特性,is_member_object_pointer,is_member_function_pointer)

类型特性 类型特性定义一个编译时基于模板的结构&#xff0c;以查询或修改类型的属性。 试图特化定义于 <type_traits> 头文件的模板导致未定义行为&#xff0c;除了 std::common_type 可依照其所描述特化。 定义于<type_traits>头文件的模板可以用不完整类型实例…

IOC课程整理-11 Spring 资源管理

1. 引入动机 2. Java 标准资源管理 3. Spring 资源接口 4. Spring 内建 Resource 实现 5. Spring Resource 接口扩展 6. Spring 资源加载器 7. Spring 通配路径资源加载器 8. Spring 通配路径资源扩展 9. 依赖注入Spring Resource 10. 依赖注入 ResourceLoader 11. 面试题精选 …

前端技术知识(含八股)总结 - 持续更新中

前端技术知识&#xff08;含八股&#xff09;总结 - 持续更新中 参考文献1.HTML和CSS1.1 语义化标签1.2 CSS 选择器及优先级 / position 定位 / box-sizing 属性 / transition / 继承属性&#xff08;如字体文字类的属性大部分有继承&#xff09;/ 行内元素和块级元素 / html的…