美文网首页面试题kmp
二叉树的遍历及应用

二叉树的遍历及应用

作者: Spicy_Crayfish | 来源:发表于2016-07-03 22:35 被阅读273次

-先序遍历:

访问根结点,先序遍历其左子树,先序遍历其右子树;运用到递归
void PreOrderTraversal ( BinTree BT )
{
if ( BT )
{
printf ( “%d”, BT -> Data );
PreOrderTraversal ( BT -> Left );
PreOrderTraversal ( BT -> Right );
}
}

-中序遍历:

中序遍历其左子树,访问根结点,中序遍历其右子树
void InOrderTraversal ( BinTree BT )
{
if ( BT )
{
InOrderTraversal ( BT -> Left );
printf ( “%d”, BT -> Data );
InOrderTraversal ( BT -> Right );
}
}

-后序遍历:

后序遍历其左子树,后序遍历其右子树,访问根结点
void PostOrderTraversal ( BinTree BT )
{
if ( BT )
{
PostOrderTraversal ( BT -> Left );
PostOrderTraversal ( BT -> Right );
printf ( “%d”, BT -> Data );
}
}

三种遍历过程中经过的结点的路线一样,只是访问各结点的时机不同,先序中序后序分别对应第一次访问时printfData第二次访问printfData和第三访问printfData

上面三种方法都是用递归,而递归根本实现方法是堆栈,现在我们直接用堆栈来实现非递归的遍历算法

eg:中序遍历非递归遍历算法:
1.遇到一个结点 就把它压栈 并去遍历它的左子树
2.当左子树遍历结束后 从栈顶弹出这个结点并访问它
3.然后按其右指针再去中序遍历该结点的右子树

void InOrderTraversal( BinTree BT ){
Stack s = CreateStack ( MaxSize ) // 创建并初始化堆栈s
while( T | | !IsEmpty(s) ){
while(T){ // 一直向左并将沿途结点压入堆栈
Push ( S, T );
T = T -> Left;
}
if ( !IsEmpty(s) ){
T = Pop(s); // 结点弹出堆栈
printf(“]”, T -> Data); //(访问)打印结点
T = T -> Right; //转向右子树
}
}

-层序遍历:

二叉树遍历的核心问题:二维结构的线性化
一个结点有两个关联(左儿子和右儿子)通过一个结点访问到一个关联结点(左儿子)后另一个关联结点(右儿子)怎么办?这个结点本身怎么办?这就需要将二维结构线性化。
解决方案:需要一个存储结构保存暂时不访问的结点 存储结构:堆栈、队列

队列实现:遍历从根结点开始 首先将根结点入队 然后开始执行循环:结点出队、访问该结点、其左右儿子入队(注意队列中先进先出原则)
序列特征:一层一层访问的
步骤:
1.从队列中取出一个元素;2.访问该元素所指结点;3.若该元素所指结点的左右儿子结点非空 则将其左右儿子的指针顺序入队。
void LevelOrderTraversal( BinTree BT ){
Queue Q ; BinTree T ;
if ( !BT ) return ; //若是空树直接返回
Q = CreateQueue ( MaxSize ) ; //创建并初始化Q
AddQ( Q, BT ) ;
while( ! IsEmptyQ ( Q ) ){
T = DeleteQ ( Q ) ;
printf ( “%d\n” , T -> Data ) ; //访问取出队列的结点
if ( T -> Left ) AddQ( Q, T -> Left );
if ( T -> Right ) AddQ( Q, T -> Right );
}
}

-遍历二叉树的应用:

eg:输出二叉树中的叶子结点,利用先序遍历(中序后序类似)
void PreOrderTraversal ( BinTree BT )
{
if ( BT )
{
if( !BT -> Left && !BT -> Right )
printf ( “%d”, BT -> Data );
PreOrderTraversal ( BT -> Left );
PreOrderTraversal ( BT -> Right );
}
}

eg:求二叉树的高度
void PostOrderGetHeight ( BinTree BT )
{ int HL, HR, MaxH ;
if ( BT )
{
HL = PostOrderGetHeight ( BT -> Left ); // 求左子树的深度
HR = PostOrderGetHeight ( BT -> Right ); //求右子树的深度
MaxH = ( HL > HR ) ? HL : HR ; //取左右子树较大的深度
return ( MaxH + 1 ); //返回树的深度
}
else return 0; // 空树深度为0
}

eg:二元运算表达式树及其遍历:

002WV0d1zy70G86PztF03&690.png

先序遍历得到前缀表达式:++a*b c*+*d e f g
中缀表达式会受到运算符优先级的影响,需要在遍历时加括号
后序遍历得到后缀表达式:a b c*+d e*f+g*+

eg:由两种遍历序列确定二叉树
(必须知道中序二叉树,若只是知道前序和后序遍历 无法确定该二叉树)
已知先序和中序遍历序列 来确定一棵二叉树
步骤:
根据先序遍历序列第一个结点 确定根结点
根据根结点在中序遍历序列中分割出左右两个子序列
对左子树和右子树分别递归使用相同的方法继续分解
practice:先序序列:a bcdefghij
中序序列:cbed a hgijf
由先序序列可知根结点为a,通过中序序列可知a左边为左子树 a右边为右子树,以此类推可确定二叉树

类似地,已知后序和中序遍历序列也可以确定一棵二叉树

相关文章

  • 3.有关二叉树的算法

    1.分层遍历二叉树:宽度优先遍历 2.分层遍历应用,按层打印二叉树 3.前序遍历二叉树 4.前序遍历,迭代 5.中...

  • 数据结构与算法

    一、线性表及应用 二、线性表的链式存储结构 三、栈与栈的应用 四、哈希表与树 五、二叉树遍历的应用之分治法 六、加...

  • 二叉树遍历方式总结(递归&迭代)

    经过一周的学习,我了解到了二叉树的遍历是非常重要且容易出错的,所以对二叉树的遍历方式及实现方式做个总结 一. 遍历...

  • 二叉树的遍历及应用

    -先序遍历: 访问根结点,先序遍历其左子树,先序遍历其右子树;运用到递归void PreOrderTraversa...

  • 二叉树 基础操作

    二叉树的使用 二叉树结构 先序创建二叉树 DFS 先序遍历二叉树 中序遍历二叉树 后序遍历二叉树 BFS 层次遍历...

  • [二叉树] 树的子结构

    前言 今天找了一到二叉树遍历的应用题,加深一下对二叉树前序遍历的理解 题目 树的子结构[https://leetc...

  • 二叉树遍历

    二叉树 二叉树的存储结构 前序遍历 中序遍历 后序遍历 遍历代码 反转二叉树 深入学习二叉树 二叉树-你必须要懂!...

  • 二叉树操作

    树节点 逐行顺序解析二叉树 前序遍历二叉树 中序遍历二叉树 后序遍历二叉树 删除指定数值的节点 前序遍历顺序存储的...

  • 0102-二叉树的层次遍历

    二叉树的层次遍历 方案一 层序遍历二叉树是典型的广度优先搜索BFS的应用,但是这里稍微复杂一点的是,我们要把各个层...

  • 关于二叉树的算法题

    前序遍历中序遍历后序遍历判断是否是平衡二叉树判断是否是对称二叉树判断二叉树高度按照层遍历二叉树判断二叉树宽度

网友评论

    本文标题:二叉树的遍历及应用

    本文链接:https://www.haomeiwen.com/subject/zxscjttx.html