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. [计算机基础]URI与URL

    URI : uniform resource identifier统一资源标示符用于指定Web资源的字符串,它定义了Web资源中的各个不同的部分.ex:http://example.org/absol ...

  2. 【设计模式】Singleton模式C++实现

    Singleton是设计模式中比较简单的一个.园中的朋友们应该都很熟悉了.前段时间参加xxx外企的面试,和面试官讨论C++的时候正好写了一个.当时由于在有些地方考虑不太周全,代码出现了一些疏漏.不过最 ...

  3. OleContainer操作Excel以二进制方式读写数据库

    需求源头:OleContainer操作Excel,想把Excel以二进制方式存入数据库,并且以二进制方式读取存入流:Procedure SaveToStream(ADOTable1: TAdoTabl ...

  4. C++历史

    C++历史 早期C++ •1979: 首次实现引入类的C(C with Classes first implemented) 1.新特性:类.成员函数.继承类.独立编译.公共和私有访问控制.友元.函数 ...

  5. [C++]C++中的运行时类型检测

    Date:2014-1-3 Summary: 使用C++中的运行时类型检测.(文章重点在于记录本人的使用情况,并非深层讨论RTTI) Contents:写习惯C#的我,在C++依然存在哪些.NET的惯 ...

  6. linux下自助获取帮助

    几乎所有命令都可以用 -h 或者--help参数 获取命令的使用方法帮助 man ls   也可以获取ls命令的详细帮助信息 man -k XXX  可以查看包含关键字XXX的文档 还有一个更详细的命 ...

  7. Java-WebSocket 项目的研究(三) WebSocketClient 类 具体解释

    通过之前两篇文章 Java-WebSocket 项目的研究(一) Java-WebSocket类图描写叙述 Java-WebSocket 项目的研究(二) 小试身手:client连接server并发送 ...

  8. 算法起步之Dijkstra算法

    原文:算法起步之Dijkstra算法 友情提示:转载请注明出处[作者 idlear    博客:http://blog.csdn.net/idlear/article/details/19687579 ...

  9. java 获取当期时间之前几小时的时间

    Calendar expireDate = Calendar.getInstance(); expireDate.set(Calendar.HOUR_OF_DAY, expireDate.get(Ca ...

  10. Ubuntu14.04搭建android开发环境

    一 下载ADT 官方下载地址:http://developer.android.com/sdk/index.html(须要FQ或者改动host) 二 解压 1 使用终端将下载的文件解压当前文件夹下: ...