二叉树的遍历:

先序,中序,后序;

二叉树的遍历有三种常见的方法,

最简单的实现就是递归调用,

另外就是飞递归的迭代调用,

最后还有O(1)空间的morris遍历;

二叉树的结构定义:

 struct TreeNode {
int val;
TreeNode *left;
TreeNode *right;
TreeNode(int x) : val(x), left(NULL), right(NULL) {}
};

1.先序遍历:

递归:

 void preOrderRecursive(TreeNode *root) {
if (!root)
return;
cout << root->val << " ";
preOrderRecursive(root->left);
preOrderRecursive(root->right);
}

迭代:

迭代要用到栈来保存父亲结点,

先序遍历,所有访问过的结点都先输出,

先遍历当前结点和当前结点的左子树,一直到左子树的最左边的结点,

遍历过的这些结点都入栈,左孩子为空时,栈顶元素设为当前结点,出栈,

然后把当前结点设为该节点的右孩子,循环一直到当前结点为空且栈也为空。

 void preOrderIterative(TreeNode *root) {
if (!root)
return;
stack<TreeNode*> stk;
TreeNode *cur = root;
while (cur || !stk.empty()) {
while (cur) {
cout << cur->val << " ";
stk.push(cur);
cur = cur->left;
}
if (!stk.empty()) {
cur = stk.top();
stk.pop();
cur = cur->right;
}
}
}
/*
* 模拟递归
* */
void preOrderIterative1(TreeNode *root) {
if (!root)
return;
stack<TreeNode*> stk;
stk.push(root);
while (!stk.empty()) {
TreeNode* tp = stk.top();
stk.pop();
cout << tp->val << " ";
if (tp->right)
stk.push(tp->right);
if (tp->left)
stk.push(tp->left);
}
}

Morris方法:

1.当前结点的左孩子为空,输出当前结点,并设置当前结点的右孩子为当前结点;

2.当前结点的左孩子不为空:

  a.找到当前结点中序遍历的前驱结点,即为该节点的左孩子的最右边的结点,前驱结点的右孩子为空,则设置前驱结点的右孩子为当前结点,并输出当前结点,设当前结点的左孩子为当前结点;

  b.前驱结点的右孩子为当前结点,则恢复前驱结点的右孩子为NULL,设当前结点的右孩子为当前结点;

 void preOrderMorris(TreeNode *root) {
if (!root)
return;
TreeNode* cur = root;
TreeNode* pre = NULL;
while (cur) {
if (cur->left == NULL) {
cout << cur->val << " ";
cur = cur->right;
}
else {
pre = cur->left;
while (pre->right != NULL && pre->right != cur)
pre = pre->right;
if (pre->right == NULL) {
cout << cur->val << " ";
pre->right = cur;
cur = cur->left;
}
else {
pre->right = NULL;
cur = cur->right;
}
}
}
}

2.中序遍历:

递归:

 void inOrderRecursive(TreeNode *root) {
if (!root)
return;
inOrderRecursive(root->left);
cout << root->val << " ";
inOrderRecursive(root->right);
}

迭代:

1.如果当前结点的左孩子不为空,则把当前结点的左孩子入栈,知道当前结点的左孩子为空;

2.如果栈不为空,则出栈,栈顶元素为当前结点,输出当前结点,并把当前结点的右孩子设为当前结点;

重复1,2直到当前结点为NULL且栈也为空;

 void inOrderIterative(TreeNode *root) {
if (!root)
return;
stack<TreeNode*> stk;
TreeNode *cur = root;
while (cur || !stk.empty()) {
while (cur) {
stk.push(cur);
cur = cur->left;
}
if (!stk.empty()) {
cur = stk.top();
stk.pop();
cout << cur->val << " ";
cur = cur->right;
}
}
}

Morris方法:

和先序遍历的过程类似,只不过输出结点的位置不一样,中序遍历是在2.b中,也就是前驱结点的右孩子为当前结点时,即当前结点的左子树都已经遍历完成时,输出当前结点;

 void inOrderMorris(TreeNode *root) {
if (!root)
return;
TreeNode* cur = root;
TreeNode* pre = NULL;
while (cur) {
if (cur->left == NULL) {
cout << cur->val << " ";
cur = cur->right;
}
else {
pre = cur->left;
while (pre->right != NULL && pre->right != cur)
pre = pre->right;
if (pre->right == NULL) {
pre->right = cur;
cur = cur->left;
}
else {
cout << cur->val << " ";
pre->right = NULL;
cur = cur->right;
}
}
}
}

3.后序遍历:

递归:

 void postOrderRecursive(TreeNode *root) {
if (!root)
return;
postOrderRecursive(root->left);
postOrderRecursive(root->right);
cout << root->val << " ";
}

迭代:

后序遍历比先序、中序都要复杂,

第一种迭代方法,可以用两个栈来模拟递归的遍历;

栈1初始化时把根节点入栈,

1.栈1出栈,把出栈的元素加入栈2,然后把该元素的左孩子(如果不为空),右孩子(如果不为空)加入栈1,知道栈1为空;

2.栈2出栈直到空,每次出栈时输出栈顶元素;

通过两个栈,保证了栈2中的元素顺序,

第二种迭代方法,

用一个结点保存访问过的最后一个结点pre,如果pre为栈顶元素的右孩子,则说明栈顶元素的右子树已经遍历过了,直接输出栈顶元素,并把当前结点设为NULL,并更新pre为出栈的元素;

1.如果当前结点存在,则一直向左遍历,入栈遍历的元素,直到结点为空;

2.栈不为空时,出栈,当前结点为栈顶元素,

如果当前结点的右孩子不为空且不为pre,说明当前结点的右子树没有遍历过,设置当前结点为该节点的右孩子,

如果右孩子为空或者为pre,直接输出当前结点,更新pre为当前结点,并设当前结点为NULL,

重复1,2直到当前结点为NULL并且栈为空;

 void postOrderIterative(TreeNode *root) {
if (!root)
return;
stack<TreeNode*> stk;
TreeNode *cur = root;
TreeNode *pre = NULL;
while (cur || !stk.empty()) {
while (cur) {
stk.push(cur);
cur = cur->left;
}
if (!stk.empty()) {
cur = stk.top();
if (cur->right != NULL && cur->right != pre) {
cur = cur->right;
}
else {
cout << cur->val << " ";
pre = cur;
stk.pop();
cur = NULL;
}
}
}
}
/*
* 双栈法
*/
void postOrderIterative1(TreeNode *root) {
if (!root)
return;
stack<TreeNode*> stk1, stk2;
TreeNode *cur;
stk1.push(root);
while (!stk1.empty()) {
cur = stk1.top();
stk1.pop();
stk2.push(cur);
if (cur->left)
stk1.push(cur->left);
if (cur->right)
stk1.push(cur->right);
}
while (!stk2.empty()) {
cout << stk2.top()->val << " ";
stk2.pop();
}
}

Morris方法:

morris方法的后序遍历较为复杂,因为需要逆序输出右孩子到父亲结点;

遍历过程与先序与中序类似,

当前驱结点的右孩子为当前结点时,左子树已经遍历完成,逆序输出当前结点的左孩子到前驱结点;

类似于链表的反转,不过反转输出之后,记得要反转回来。

 void reverse(TreeNode *begin, TreeNode *end) {
if (begin == end)
return;
TreeNode *pre = begin;
TreeNode *cur = begin->right;
TreeNode *next;
while (pre != end) {
temp = cur->right;
cur->right = pre;
pre = cur;
cur = temp;
}
} void traversalReversal(TreeNode *begin, TreeNode *end) {
reverse(begin, end);
TreeNode *it = end;
while (true) {
cout << it->val << " ";
if (it == begin)
break;
it = it->right;
}
reverse(end, begin);
} void postOrderMorris(TreeNode *root) {
if (!root)
return;
TreeNode dump();
dump.left = root;
TreeNode *cur = &dump;
TreeNode *pre = NULL;
while (cur) {
if (cur->left == NULL) {
cur = cur->right;
}
else {
pre = cur->left;
while (pre->right != NULL && pre->right != cur)
pre = pre->right;
if (pre->right == NULL) {
pre->right = cur;
cur = cur->left;
}
else {
traversalReverse(cur->left, pre);
pre->right = NULL;
cur = cur->right;
}
}
}
}

参考:

http://www.cnblogs.com/AnnieKim/archive/2013/06/15/MorrisTraversal.html

http://blog.csdn.net/hackbuteer1/article/details/6583988

二叉树的遍历(递归,迭代,Morris遍历)的更多相关文章

  1. 二叉树的建立&&前中后遍历(递归实现)&&层次遍历

    下面代码包含了二叉树的建立过程,以及三种遍历方法了递归实现,代码中还利用队列实现了层次遍历. import java.util.LinkedList; import java.util.Queue; ...

  2. LeetCode:二叉树的非递归中序遍历

    第一次动手写二叉树的,有点小激动,64行的if花了点时间,上传leetcode一次点亮~~~ /* inorder traversal binary tree */ #include <stdi ...

  3. Python实现二叉树的非递归先序遍历

    思路: 1. 使用列表保存结果: 2. 使用栈(列表实现)存储结点: 3. 当根结点存在,保存结果,根结点入栈: 4. 将根结点指向左子树: 5. 根结点不存在,栈顶元素出栈,并将根结点指向栈顶元素的 ...

  4. Python实现二叉树的非递归中序遍历

    思路: 1. 使用一个栈保存结点(列表实现): 2. 如果结点存在,入栈,然后将当前指针指向左子树,直到为空: 3. 当前结点不存在,则出栈栈顶元素,并把当前指针指向栈顶元素的右子树: 4. 栈不为空 ...

  5. Morris 遍历实现二叉树的遍历

    Morris 遍历实现二叉树的遍历 作者:Grey 原文地址: 博客园:Morris 遍历实现二叉树的遍历 CSDN:Morris 遍历实现二叉树的遍历 说明 Morris 遍历可以实现二叉树的先,中 ...

  6. 经典算法 Morris遍历

    内容: 1.什么是morris遍历 2.morris遍历规则与过程 3.先序及中序 4.后序 5.morris遍历时间复杂度分析 1.什么是morris遍历 关于二叉树先序.中序.后序遍历的递归和非递 ...

  7. 二叉树的遍历(递归,迭代,Morris遍历)

    二叉树的三种遍历方法: 先序,中序,后序,这三种遍历方式每一个都可以用递归,迭代,Morris三种形式实现,其中Morris效率最高,空间复杂度为O(1). 主要参考博客: 二叉树的遍历(递归,迭代, ...

  8. [转载]Morris Traversal方法遍历二叉树(非递归,不用栈,O(1)空间)

    本文主要解决一个问题,如何实现二叉树的前中后序遍历,有两个要求: 1. O(1)空间复杂度,即只能使用常数空间: 2. 二叉树的形状不能被破坏(中间过程允许改变其形状). 通常,实现二叉树的前序(pr ...

  9. Morris Traversal方法遍历二叉树(非递归,不用栈,O(1)空间)——无非是在传统遍历过程中修改叶子结点加入后继结点信息(传统是stack记录),然后再删除恢复

    先看看线索二叉树 n个结点的二叉链表中含有n+1(2n-(n-1)=n+1)个空指针域.利用二叉链表中的空指针域,存放指向结点在某种遍历次序下的前驱和后继结点的指针(这种附加的指针称为"线索 ...

随机推荐

  1. Centos 7: 打开Samba防火墙端口

    firewall-cmd --permanent --add-port=137/tcp firewall-cmd --permanent --add-port=138/tcp firewall-cmd ...

  2. iOS 基础日记-修饰符

    今晚随便温习了一下iOS 基础关于修饰符这块的东西,下面简单的来描述一下,其中有的也是在网络学习到的: strong与weak是由ARC新引入的对象变量属性 ARC的解释:ARC引入了新的对象的生命周 ...

  3. Android源码分析之HandlerThread

    HandlerThread是一种特殊的Thread,也就是有Looper的thread,既然有looper的话,那我们就可以用此looper来 创建一个Handler,从而实现和它的交互,比如你可以通 ...

  4. iOS 学习 - 6.Objective-C中的各种遍历(迭代)方式

    说明:转自文顶顶 一.使用 for 循环 要遍历字典.数组或者是集合,for 循环是最简单也用的比较多的方法 -(void)iteratorWithFor { //////////处理数组////// ...

  5. 小波说雨燕 第三季 构建 swift UI 之 UI组件集-视图集(五)Image View视图 学习笔记

    留下两个问题:1.后面涉及到的异常不知道原因.2.动态图片到了程序里面就不动了.       然后:   上面是有问题的,下面是没有问题的了.    代码(另外简单写的代码,纠正了那个错误): imp ...

  6. Javascript中的五种数据类型

    Undefined 未定义.只有一个值undefined Null         只有一个值,null Boolean 在javascript中,只要逻辑表达式不返回undefined不返回null ...

  7. n&(n-1) n&(-n)

    n&(n-1)   n&(-n) n&(n-1)作用:将n的二进制表示中的最低位为1的改为0,先看一个简单的例子:n = 10100(二进制),则(n-1) = 10011 = ...

  8. javascript/jquery 常见功能实现(持续更新...)

    1. input 只能输入整数数字和字母 $(document).on('keyup','#no',function(){ var val = $.trim($(this).val()); if(va ...

  9. oracle的增删改查语句

    创建一个表: cteate table 表名(列1 类型, 列2 类型);查看表结构 desc表名添加一个字段 alter table 表名 add(列类型);修改字段类型 alter table 表 ...

  10. 如何在WIN7下进行LINUX虚拟机搭建

    Linux是一套免费使用和自由传播的类Unix操作系统,非常适用于搭建网络服务器等,我本人日常工作时,是使用的LINUX和WIN7双操作系统,但每次更换系统总要关机重启很不方便,所以也在WIN7下搭建 ...