import java.util.Stack;

//二叉树3种递归和非递归遍历(Java)
public class Traverse { /******************一二进制树的定义**************************/
private final int MAX_SIZE = 10;
//链式存储
public static class BinaryTreeNode
{
int mValue;
BinaryTreeNode mLeft;
BinaryTreeNode mRight; public BinaryTreeNode(int mValue) {
this.mValue = mValue;
}
} //顺序存储
class BinaryTreeNode2
{
int[] data = new int[MAX_SIZE];
int length;
} //用以实现兴许遍历的辅助结构
private class HelpNode
{
BinaryTreeNode treeNode;
boolean isFirst;
} /******************递归实现***************************/
//先序遍历
public int PreOrderTreeWalk(BinaryTreeNode pNode)
{
if(pNode == null)
return 0;
visitNode(pNode);
PreOrderTreeWalk(pNode.mLeft);
PreOrderTreeWalk(pNode.mRight);
return 1;
} //中序遍历
public int InOrderTreeWalk(BinaryTreeNode pNode)
{
if(pNode == null)
return 0;
InOrderTreeWalk(pNode.mLeft);
visitNode(pNode);
InOrderTreeWalk(pNode.mRight);
return 1;
} //后序遍历
public int PostOrderTreeWalk(BinaryTreeNode pNode)
{
if(pNode == null)
return 0;
PostOrderTreeWalk(pNode.mLeft);
PostOrderTreeWalk(pNode.mRight);
visitNode(pNode);
return 1;
} /*****************非递归实现***********************/
//先序遍历
public int PreOrderTraverse(BinaryTreeNode pNode)
{
Stack<BinaryTreeNode> stack = new Stack<>();
if(pNode == null)
return 0; while(!stack.isEmpty()||pNode != null)
{
while(pNode != null)
{
//先訪问
visitNode(pNode);
stack.push(pNode);
//遍历左节点
pNode = pNode.mLeft;
}
//返回顶层元素
pNode = stack.peek();
stack.pop();
//遍历右节点
pNode = pNode.mRight;
}
return 1;
} //先序遍历实现方法二
public int PreOrderTraverse2(BinaryTreeNode pNode)
{
if(pNode == null)
return 0;
Stack<BinaryTreeNode> stack = new Stack<>();
stack.push(pNode); while(!stack.isEmpty())
{
pNode = stack.pop();
visitNode(pNode); if(pNode.mRight != null)
stack.push(pNode.mRight);
if(pNode.mLeft != null)
stack.push(pNode.mLeft);
}
return 1;
} //中序遍历
public int InOrderTraverse(BinaryTreeNode pNode)
{
Stack<BinaryTreeNode> stack = new Stack<>();
if(pNode == null)
return 0; while(!stack.isEmpty()||pNode != null)
{
while(pNode!=null)
{
stack.push(pNode);
pNode = pNode.mLeft;
}
pNode = stack.pop();
visitNode(pNode);
pNode = pNode.mRight;
}
return 1;
} //后序遍历,用一个标记标记右子树是否訪问过
/*
* 第一种思路:对于任一结点P,将其入栈,然后沿其左子树一直往下搜索,直到搜索到没有左孩子的结点,
* 此时该结点出如今栈顶。可是此时不能将其出栈并訪问。因此其右孩子还为被訪问。 所以接下来依照同样
* 的规则对其右子树进行同样的处理。当訪问完其右孩子时。该结点又出如今栈顶,此时能够将其出栈并訪
* 问。这样就保证了正确的訪问顺序。能够看出。在这个过程中。每个结点都两次出如今栈顶,仅仅有在第二
* 次出如今栈顶时,才干訪问它。 因此须要多设置一个变量标识该结点是否是第一次出如今栈顶。
* */
public int PostOrderTraverse(BinaryTreeNode pNode)
{
if(pNode == null)
return 0;
Stack<HelpNode> stack = new Stack<>();
HelpNode helpNode;
while(!stack.isEmpty() || pNode != null)
{
//一直循环至最左节点
while(pNode != null)
{
HelpNode temp = new HelpNode();
temp.treeNode = pNode;
temp.isFirst = true;
stack.push(temp);
pNode = pNode.mLeft;
} if(!stack.isEmpty())
{
helpNode = stack.pop(); if(helpNode.isFirst)//表示第一次,即每个要被訪问的根节点要被push两次
{
helpNode.isFirst = false;
stack.push(helpNode);
pNode = helpNode.treeNode.mRight;//右节点的是否有效则移至循环的開始出进行推断
}
else
{
visitNode(helpNode.treeNode);
pNode = null;
}
}
}
return 1;
} //后序遍历实现方法二:双栈法
public int PostOrderTraverse2(BinaryTreeNode pNode)
{
if(pNode == null)
return 0;
Stack<BinaryTreeNode> stack1 = new Stack<>();
Stack<BinaryTreeNode> stack2 = new Stack<>();//辅助栈
//存入根节点,初始化
stack1.push(pNode);
//stack1弹出的元素,压入stack2,在将该元素的左右节点压入stack1
while(!stack1.isEmpty())
{
pNode = stack1.pop();
stack2.push(pNode);
if(pNode.mLeft != null)
{
stack1.push(pNode.mLeft);
}
if(pNode.mRight != null)
{
stack1.push(pNode.mRight);
}
} //stack弹出的即是后序遍历的顺序
while(!stack2.isEmpty())
{
visitNode(stack2.pop());
}
return 1;
} //后序遍历实现方法三
/*
* 另外一种思路:要保证根结点在左孩子和右孩子訪问之后才干訪问,因此对于任一结点P,先将其入栈。
* 假设P不存在左孩子和右孩子,则能够直接訪问它。或者P存在左孩子或者右孩子,可是其左孩子和右
* 孩子都已被訪问过了。则同样能够直接訪问该结点。若非上述两种情况。则将P的右孩子和左孩子依次
* 入栈,这样就保证了每次取栈顶元素的时候,左孩子在右孩子前面被訪问,左孩子和右孩子都在根结点
* 前面被訪问。
* */
//后序遍历实现方法二:双栈法
public int PostOrderTraverse3(BinaryTreeNode pNode)
{
if(pNode == null)
return 0; BinaryTreeNode preVisitedNode = null;
Stack<BinaryTreeNode> stack = new Stack<>();
stack.push(pNode); while(!stack.isEmpty())
{
pNode = stack.peek();
if((pNode.mLeft == null && pNode.mLeft == null)//左右子树均为空的情况,即叶子节点
||(preVisitedNode != null &&
(preVisitedNode == pNode.mLeft || preVisitedNode == pNode.mRight)))//左右子树已经被訪问的情况
{
visitNode(pNode);
preVisitedNode = stack.pop();
}
else
{
if(pNode.mRight != null)
stack.push(pNode.mRight);//注意push的顺序,先訪问右子树
if(pNode.mLeft != null)
stack.push(pNode.mLeft);
}
} return 1;
} /*********辅助函数**********/
//訪问节点数据
private void visitNode(BinaryTreeNode treeNode)
{
System.out.print(treeNode.mValue);
System.out.print("、");
} public static void main(String[] args)
{
/************************构造測试二叉树链表组***********************/
int[] data = {1,2,3,4,5,6,7,8,9,10};
BinaryTreeNode[] treeNodes = new BinaryTreeNode[data.length]; for(int i = 0; i < data.length; i++)
{
treeNodes[i] = new BinaryTreeNode(data[i]);
} for(int i = 0; i < (data.length/2); i ++)
{
treeNodes[i].mLeft = treeNodes[(i + 1)*2 - 1]; if(((i + 1) *2) < data.length)
{
treeNodes[i].mRight = treeNodes[(i + 1)*2];
}
}
Traverse traverse = new Traverse();
BinaryTreeNode root = treeNodes[0];
System.out.print("先序遍历递归: ");
traverse.PreOrderTreeWalk(root);
System.out.println(); System.out.print("先序遍历非递归一:");
traverse.PreOrderTraverse(root);
System.out.println(); System.out.print("先序遍历非递归二:");
traverse.PreOrderTraverse2(root);
System.out.println(); System.out.print("中序遍历递归: ");
traverse.InOrderTreeWalk(root);
System.out.println(); System.out.print("中序遍历递归: ");
traverse.InOrderTraverse(root);
System.out.println(); System.out.print("后序遍历递归: ");
traverse.PostOrderTreeWalk(root);
System.out.println(); System.out.print("后序遍历非递归一:");
traverse.PostOrderTraverse(root);
System.out.println(); System.out.print("后序遍历非递归二:");
traverse.PostOrderTraverse2(root);
System.out.println(); System.out.print("后序遍历非递归三:");
traverse.PostOrderTraverse3(root);
System.out.println();
/******************验证输出***********************/
// for(BinaryTreeNode treeNode : treeNodes)
// {
// System.out.print("根节点");
// System.out.println(treeNode.mValue);
// System.out.print("左节点");
// if(treeNode.mLeft != null)
// {
// System.out.println(treeNode.mLeft.mValue);
// }
// else
// {
// System.out.println("null");
// }
//
// System.out.print("右节点");
// if(treeNode.mRight != null)
// {
// System.out.println(treeNode.mRight.mValue);
// }
// else
// {
// System.out.println("null");
// }
// System.out.println();
// }
} }

版权声明:本文博主原创文章。博客,未经同意不得转载。

二叉树3种递归和非递归遍历(Java)的更多相关文章

  1. 数据结构二叉树的递归与非递归遍历之java,javascript,php实现可编译(1)java

    前一段时间,学习数据结构的各种算法,概念不难理解,只是被C++的指针给弄的犯糊涂,于是用java,web,javascript,分别去实现数据结构的各种算法. 二叉树的遍历,本分享只是以二叉树中的先序 ...

  2. C实现二叉树(模块化集成,遍历的递归与非递归实现)

    C实现二叉树模块化集成 实验源码介绍(源代码的总体介绍):header.h : 头文件链栈,循环队列,二叉树的结构声明和相关函数的声明.LinkStack.c : 链栈的相关操作函数定义.Queue. ...

  3. JAVA递归、非递归遍历二叉树(转)

    原文链接: JAVA递归.非递归遍历二叉树 import java.util.Stack; import java.util.HashMap; public class BinTree { priva ...

  4. 二叉树前中后/层次遍历的递归与非递归形式(c++)

    /* 二叉树前中后/层次遍历的递归与非递归形式 */ //*************** void preOrder1(BinaryTreeNode* pRoot) { if(pRoot==NULL) ...

  5. 二叉树之AVL树的平衡实现(递归与非递归)

    这篇文章用来复习AVL的平衡操作,分别会介绍其旋转操作的递归与非递归实现,但是最终带有插入示例的版本会以递归呈现. 下面这张图绘制了需要旋转操作的8种情况.(我要给做这张图的兄弟一个赞)后面会给出这八 ...

  6. 数据结构-树以及深度、广度优先遍历(递归和非递归,python实现)

    前面我们介绍了队列.堆栈.链表,你亲自动手实践了吗?今天我们来到了树的部分,树在数据结构中是非常重要的一部分,树的应用有很多很多,树的种类也有很多很多,今天我们就先来创建一个普通的树.其他各种各样的树 ...

  7. java扫描文件夹下面的所有文件(递归与非递归实现)

    java中扫描指定文件夹下面的所有文件扫描一个文件夹下面的所有文件,因为文件夹的层数没有限制可能多达几十层几百层,通常会采用两种方式来遍历指定文件夹下面的所有文件.递归方式非递归方式(采用队列或者栈实 ...

  8. 回溯算法 DFS深度优先搜索 (递归与非递归实现)

    回溯法是一种选优搜索法(试探法),被称为通用的解题方法,这种方法适用于解一些组合数相当大的问题.通过剪枝(约束+限界)可以大幅减少解决问题的计算量(搜索量). 基本思想 将n元问题P的状态空间E表示成 ...

  9. C语言实现 二分查找数组中的Key值(递归和非递归)

    基本问题:使用二分查找的方式,对数组内的值进行匹配,如果成功,返回其下标,否则返回 -1.请使用递归和非递归两种方法说明. 非递归代码如下: #include <stdio.h> int ...

随机推荐

  1. Mysql对自增主键ID进行重新排序

    Mysql数据库表的自增主键ID号经过一段时间的添加与删除之后乱了,需要重新排列. 原理:删除原有的自增ID,重新建立新的自增ID. 1,删除原有主键: ALTER TABLE `table_name ...

  2. 网络数据(socket)传输总结

    环境限定:TCP/IP下的socket网络传输:C/C++开发语言,32/64位机. 目前有两种方式对数据进行传输:1)字符流形式,即将数据用字符串表示:2)结构型方式,即将数据按类型直接传输. 1) ...

  3. MySql安装(rpm)和启动配置

    MySql安装(rpm)和启动配置 安装环境: OS:Oracle Linux 5.9 安装步骤: 1.解压MySql安装包 [root@bakdbserver mysql]# tar -xf MyS ...

  4. python中的中文编码

    我现在编写python代码,有一些内容需要用中文编写,例如注释,一些其它的东西 默认python是不支持中文的,包括两个方面不支持,一是文件编码默认是ansi的,二是虚拟机运行解析脚本时也是非utf的 ...

  5. Android改变系统自带环形ProgressBar的大小

    MainActivity如下: package cc.testprogressbar; import android.os.Bundle; import android.app.Activity; / ...

  6. IDFA的值什么时候会发生改变

    在何种情况下 , 应用的IDFA值会发生改变? 近期工作中须要获得一个能够唯一地标示每个不同应用的ID,之前的苹果UDID已经不让使用了. 那么我们须要使用新的IDFA来引用.可是在某些情况下这个ID ...

  7. 奔小康赚大钱 hdu 2255

    奔小康赚大钱 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  8. 详谈隐藏Tabbar的几种方法

    如今正在写的一个项目,涉及到了使用两个TabBar,然后我须要显示当中一个的时候,然后隐藏另外一个,可是中间却出现故障了.我查了一些资料,想总结一下关于TabBar的隐藏. 第一种方法是: //隐藏t ...

  9. sqlplus登录、连接命令

    经常使用: sqlplus username/password  如:普通用户登录  sqlplus scott/tiger sqlplus username/password@net_service ...

  10. HDU3257 Hello World!

    Hello World! Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tot ...