二叉树先序遍历的非递归算法

article/2025/9/1 18:14:12

在前面一文,说过二叉树的递归遍历算法(二叉树先根(先序)遍历的改进),此文主要讲二叉树的非递归算法,采用栈结构

总结先根遍历得到的非递归算法思想如下:

1)入栈,主要是先头结点入栈,然后visit此结点

2)while,循环遍历当前结点,直至左孩子没有结点

3)if结点的右孩子为真,转入1)继续遍历,否则退出当前结点转入父母结点遍历转入1)

先看符合此思想的算法:

int PreOrderTraverseNonRecursiveEx(const BiTree &T, int (*VisitNode)(TElemType data))
{if (T == NULL){return -1;}BiTNode *pBiNode = T;SqStack S;InitStack(&S);Push(&S, (SElemType)T);while (!IsStackEmpty(S)){while (pBiNode){VisitNode(pBiNode->data);if (pBiNode != T){Push(&S, (SElemType)pBiNode);}			pBiNode = pBiNode->lchild;}if(pBiNode == NULL){Pop(&S, (SElemType*)&pBiNode); }		if ( pBiNode->rchild == NULL){Pop(&S, (SElemType*)&pBiNode); //如果此时栈已空,就有问题}pBiNode = pBiNode->rchild;}return 0;
}

注意:1)这里使用了栈结构,可参看上文顺序结构存储的栈

            2)这里在保存结点的时候,我保存的是指针也就是结点的地址,将其变为int型存储,在pop的时候里面使用的是指针,所以取的是&pBiNode,而不是pBiNode,为什么请自行思考指针的使用,最好理解的就是BiTNode *pBiNode;定义改为BiTree pBiNode就很好理解了。

上面这个算法其实是错误的!为什么呢? 这里我检查好久,期间出现还出现过无限循环,也出现过从左子树退出后右边子树不显示,最后我修改了第一个while判断条件,为什么呢?因为如果在pop之后,栈已空但是右子树还有,就无法继续了,这个在我写出后并没有进行太多验证,后面再阐述,这里并没有压入null指针,看一下压入空指针的例子,主要是左子树为空的时候才压入栈的,如下:

int PreOrderTraverseNonRecursive(const BiTree &T, int (*VisitNode)(TElemType data))
{if (T == NULL){return -1;}BiTNode *pBiNode = T;SqStack S;InitStack(&S);Push(&S, (SElemType)T);while (!IsStackEmpty(S)){GetTop(S, (SElemType*)&pBiNode);while (pBiNode){VisitNode(pBiNode->data);		pBiNode = pBiNode->lchild;Push(&S, (SElemType)pBiNode);}if(pBiNode == NULL){ Pop(&S, (SElemType*)&pBiNode);}		if ( !IsStackEmpty(S)){Pop(&S, (SElemType*)&pBiNode);pBiNode = pBiNode->rchild;Push(&S, (SElemType)pBiNode);}}return 0;
}
这里是这样的,先压入根节点,然后判断左子树是否为空,不为空就压入栈,否则退出while循环之后就将NULL结点出栈,再判断当前栈是否为空,如果非空就出栈得到父节点然后判断右孩子,压入右孩子结点,再判断此右子树的左孩子是否为空,继续循环。

这里有两个浪费的地方:一个就是压入空孩子结点入栈,二就是频繁使用GetTop获得栈顶元素

这里返回过来再看初开始设计的算法,那里正好没有压入NULL指针或者说空的孩子结点,但是并不能输出完整,这里我们想到可以在判断栈的时候加入,当前的结点是否为NULL就可以了,这样就不会出现不会显示退出左子树结点不能显示右子树结点的尴尬了,如下:

//非递归先序遍历二叉树
int PreOrderTraverseNonRecursiveEx(const BiTree &T, int (*VisitNode)(TElemType data))
{if (T == NULL){return -1;}BiTNode *pBiNode = T;SqStack S;InitStack(&S);Push(&S, (SElemType)T);while ( !IsStackEmpty(S) || pBiNode)  //主要修改的就是这句{while (pBiNode){VisitNode(pBiNode->data);if (pBiNode != T){Push(&S, (SElemType)pBiNode);}			pBiNode = pBiNode->lchild;}if(pBiNode == NULL){Pop(&S, (SElemType*)&pBiNode); }		if ( pBiNode->rchild == NULL){Pop(&S, (SElemType*)&pBiNode); }pBiNode = pBiNode->rchild;}return 0;
}
在第一个while循环加入这个之后,就可以了,测试用例与二叉树先序遍历类似。如下测试上节的二叉树例子:


此时输入的数据仍然还是 12 34 0 0 78 0 0,测试结果如下:

--- BiTree ---
Please Enter BiTree Node data:
12
Please Enter BiTree Node data:
34
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
78
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
0
12 34 78

这个还不足以测试,再看如下的二叉树

此时输入数据应该为:12 34 24 0 0 50 0 0 78 37 0 0 0,测试结果如下:

--- BiTree ---
Please Enter BiTree Node data:
12
Please Enter BiTree Node data:
34
Please Enter BiTree Node data:
24
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
50
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
78
Please Enter BiTree Node data:
37
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
0
Please Enter BiTree Node data:
0
12 34 24 50 78 37

由先序遍历可知,正好是正确的,另外这些算法不光是对先序遍历的,如果想变为中序或者后序,只需将上面算法中的visit之类的先去掉,然后将它加入合适的位置,就可以了


http://chatgpt.dhexx.cn/article/KcFU1G9O.shtml

相关文章

二叉树的先序递归遍历算法

怎样遍历一棵二叉树呢?把它看成三个部分: 根结点,左子树,右子树,所以要遍历一棵二叉树,就要分别遍历这三个部分 访问完毕左子树 访问完毕右子树 序列: 如何实现算法?首先考虑存储结构…

关于二叉树先序遍历和后序遍历为什么不能唯一确定一个二叉树分析

文章目录 二叉树唯一确定先序和中序递归构建二叉树的过程先序和后序递归构建二叉树的过程先序和后序递归构建二叉树的代码如果二叉树不唯一,怎么处理完整代码分析自己的问题 二叉树唯一确定 对于一个二叉树,并不是说给出了先序和后序就是无法唯一确定的。…

二叉树的递归遍历(先序输入)

今天来看看二叉树的递归遍历&#xff0c;我们要实现二叉树的先序&#xff0c;中序&#xff0c;后续遍历 这里我们采用的是先序输入 下面是完整代码 #include <stdio.h> #include <stdlib.h> struct node {char data;node *Lchild;node *Rchild; }tree; node *cr…

根据二叉树先序遍历和中序遍历构建二叉树

前方有一个人在等着你&#xff0c;你只管勇敢的向前走 采用递归分治的思想&#xff0c;将一个大问题划分成子问题&#xff0c; 对于本题&#xff0c;根据二叉树先序遍历和中序遍历构建二叉树&#xff0c;思路&#xff1a; 我们可以求得根节点左子树的先序和中序序列&#xff0…

二叉树前序、中序、后序遍历非递归写法的透彻解析

前言 在前两篇文章二叉树和二叉搜索树中已经涉及到了二叉树的三种遍历。递归写法&#xff0c;只要理解思想&#xff0c;几行代码。可是非递归写法却很不容易。这里特地总结下&#xff0c;透彻解析它们的非递归写法。其中&#xff0c;中序遍历的非递归写法最简单&#xff0c;后…

二叉树的后序遍历

二叉树文章系列&#xff1a; 二叉树的前序遍历二叉树的中序遍历二叉树的后序遍历二叉树的层序遍历二叉树的前序、中序、后序、层序遍历【解法完整版】 本文目录 一、解题思路&#xff1a;递归二、解题思路&#xff1a;迭代&#xff08;方法1&#xff09;三、解题思路&#xff…

C语言完整代码实现:二叉树的先序遍历、中序遍历、后序遍历

一、先序遍历原理 先序遍历就是&#xff1a;根、左、右&#xff0c;也就是先遍历根结点再遍历左结点最后再遍历右结点&#xff0c;注意&#xff1a;如果遍历到的结点不是叶子结点的话需要对该结点进行拆分&#xff0c;比如这棵二叉树&#xff1a; 先遍历A&#xff0c;然后是B&a…

数据结构——二叉树的先序遍历

二叉树的遍历分为 先序遍历&#xff0c;中序遍历&#xff0c;后序遍历&#xff0c;层次遍历 四种遍历。 这节要分享的是先序遍历 如图所示&#xff0c;这是一个普通的二叉树。他的先序遍历是&#xff1a;A B D E H C F G I J 为什么呢&#xff1f; 先序遍历的遍历规则是&am…

二叉树三种遍历顺序

三.二叉树的三种遍历方式 1.先序遍历&#xff1a;按照根节点->左子树->右子树的顺序访问二叉树 先序遍历&#xff1a;&#xff08;1&#xff09;访问根节点&#xff1b;&#xff08;2&#xff09;采用先序递归遍历左子树&#xff1b;&#xff08;3&#xff09;采用先序…

二叉树(Binary Tree):先序遍历、中序遍历、后序遍历和层次遍历

二叉树&#xff08;Binary Tree&#xff09;&#xff1a;先序遍历、中序遍历、后序遍历和层次遍历 树 Tree二叉树 Binary Tree先序遍历 Preorder Traversal中序遍历 Inoreder Traversal后序遍历 Postorder Traversal层次遍历 Level Traversal 树 Tree 根 Root&#xff1a;树顶部…

oracle awr监控报告,一个Oracle小白的AWR报告分析(一)

背景&#xff1a;某个类似准实时的数据分析系统&#xff0c;每15分钟从其他6个数据库中抽取五百张增量数据表&#xff0c;并进行15分钟粒度统计&#xff0c;同时有个前端门户进行查询。 该数据分析系统由数据抽取服务器、应用服务器、数据库服务器组成&#xff0c;全部为虚拟机…

oracle生成awr报告命令,Oracle AWR报告生成方法

1、登录Oracle程序所在的服务器&#xff0c;查找出awrrpt.sql文件所在位置 D:\oracle\product\10.2.0\db_1\RDBMS\ADMIN\awrrpt.sql 2、登录Oracle&#xff0c;以sysdba身份连接 3、执行命令 D:\oracle\product\10.2.0\db_1\RDBMS\ADMIN\awrrpt.sql 4、输入report_type报告类型…

oracle打印awr报告,oracle导出AWR报告步骤

1、进入数据库 sqlplus / as sysdba ps:如果出现用户密码错误&#xff0c; 计算机管理 > 组 > ora_dba组里的用户登陆操作系统&#xff0c;就可以无需输入用户和口令&#xff0c;直接以sysdba的身份连上数据库。 2、查看用户 show parameter db_name 3、开始压测后执行 e…

Oracle导出AWR报告

一、使用root用户登录Linux服务器 二、切换至oracle用户 执行命令&#xff1a;su – oracle&#xff0c;然后回车 三、使用管理员权限连接数据库 执行命令&#xff1a;sqlplus / as sysdba&#xff0c;然后回车 四、生成报告快照 执行脚本&#xff1a;exec DBMS_WORKLOAD_RE…

如何分析AWR 报告

Automatic Workload Repository是10g引入的一个重要组件。在里面存贮着近期一段时间内&#xff0c;默认是7天&#xff0c;数据库活动状态的详细信息。 AWR报告是对AWR视图进行查询而得到的一份自动生成的报告。可以通过下面的脚本手工得到一份AWR报告。 exec dbms_w…

oracle 取awr报告,Oracle生成awr报告

Oracle生成awr报告 达芬奇的梦 2018-04-22 21:28:32 Oracle 一、手工生成awr报告的方法 1、相应权限用户登录(sysdba)后,在$ORACLE_HOME/rdbms/admin 2、在sqlplus里执行@?/rdbms/admin/awrrpt.sql,按照提示操作。 3、生成AWR报告说明 单实例:@$ORACLE_HOME/rdbms/admin/aw…

Oracle SQL调优系列之AWR报告简介

文章目录 一、AWE报告生成步骤1.1 工具选择1.2 自动创建快照1.3 手工创建快照1.4 生成AWR报告 二、AWR报告分析2.1 AWR之DB Time2.2 AWR之load_profile2.3 AWR之efficiency percentages2.4 AWR之top 10 events2.5 AWR之SQL Statistics 一、AWE报告生成步骤 对于SQL调优&#x…

AWR报告解读

0 初步结论 ① 数据库CPU资源不够&#xff0c;CPU使用率较高&#xff0c;造成CPU等待时间较长&#xff0c;可适当提升CPU资源&#xff1b; ② 数据库I/O资源消耗不太大&#xff0c;不存在IO瓶颈&#xff1b; ③ 可适当调大SGA空间&#xff08;增加10G左右&#xff09;&#xf…

用sql统计vintage,滚动率,迁移率,逾期率

获取代码请移步&#xff1a;用sql统计vintage&#xff0c;滚动率&#xff0c;迁移率&#xff0c;逾期率

如何用R语言做Vintage分析

一、背景 Vintage一词源自葡萄酒业&#xff0c;意思是葡萄酒酿造年份。因为每年的天气、温度、湿度、病虫害等情况不同&#xff0c;而这些因素都会对葡萄酒的品质产生很大的影响&#xff0c;所以人们对葡萄酒以葡萄当年的采摘年份进行标识来加以品质区分。现在Vintage分析被广泛…