二叉树的遍历(递归,迭代,Morris遍历)
二叉树的遍历:
先序,中序,后序;
二叉树的遍历有三种常见的方法,
最简单的实现就是递归调用,
另外就是飞递归的迭代调用,
最后还有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遍历)的更多相关文章
- 二叉树的建立&&前中后遍历(递归实现)&&层次遍历
下面代码包含了二叉树的建立过程,以及三种遍历方法了递归实现,代码中还利用队列实现了层次遍历. import java.util.LinkedList; import java.util.Queue; ...
- LeetCode:二叉树的非递归中序遍历
第一次动手写二叉树的,有点小激动,64行的if花了点时间,上传leetcode一次点亮~~~ /* inorder traversal binary tree */ #include <stdi ...
- Python实现二叉树的非递归先序遍历
思路: 1. 使用列表保存结果: 2. 使用栈(列表实现)存储结点: 3. 当根结点存在,保存结果,根结点入栈: 4. 将根结点指向左子树: 5. 根结点不存在,栈顶元素出栈,并将根结点指向栈顶元素的 ...
- Python实现二叉树的非递归中序遍历
思路: 1. 使用一个栈保存结点(列表实现): 2. 如果结点存在,入栈,然后将当前指针指向左子树,直到为空: 3. 当前结点不存在,则出栈栈顶元素,并把当前指针指向栈顶元素的右子树: 4. 栈不为空 ...
- Morris 遍历实现二叉树的遍历
Morris 遍历实现二叉树的遍历 作者:Grey 原文地址: 博客园:Morris 遍历实现二叉树的遍历 CSDN:Morris 遍历实现二叉树的遍历 说明 Morris 遍历可以实现二叉树的先,中 ...
- 经典算法 Morris遍历
内容: 1.什么是morris遍历 2.morris遍历规则与过程 3.先序及中序 4.后序 5.morris遍历时间复杂度分析 1.什么是morris遍历 关于二叉树先序.中序.后序遍历的递归和非递 ...
- 二叉树的遍历(递归,迭代,Morris遍历)
二叉树的三种遍历方法: 先序,中序,后序,这三种遍历方式每一个都可以用递归,迭代,Morris三种形式实现,其中Morris效率最高,空间复杂度为O(1). 主要参考博客: 二叉树的遍历(递归,迭代, ...
- [转载]Morris Traversal方法遍历二叉树(非递归,不用栈,O(1)空间)
本文主要解决一个问题,如何实现二叉树的前中后序遍历,有两个要求: 1. O(1)空间复杂度,即只能使用常数空间: 2. 二叉树的形状不能被破坏(中间过程允许改变其形状). 通常,实现二叉树的前序(pr ...
- Morris Traversal方法遍历二叉树(非递归,不用栈,O(1)空间)——无非是在传统遍历过程中修改叶子结点加入后继结点信息(传统是stack记录),然后再删除恢复
先看看线索二叉树 n个结点的二叉链表中含有n+1(2n-(n-1)=n+1)个空指针域.利用二叉链表中的空指针域,存放指向结点在某种遍历次序下的前驱和后继结点的指针(这种附加的指针称为"线索 ...
随机推荐
- Centos 7: 打开Samba防火墙端口
firewall-cmd --permanent --add-port=137/tcp firewall-cmd --permanent --add-port=138/tcp firewall-cmd ...
- iOS 基础日记-修饰符
今晚随便温习了一下iOS 基础关于修饰符这块的东西,下面简单的来描述一下,其中有的也是在网络学习到的: strong与weak是由ARC新引入的对象变量属性 ARC的解释:ARC引入了新的对象的生命周 ...
- Android源码分析之HandlerThread
HandlerThread是一种特殊的Thread,也就是有Looper的thread,既然有looper的话,那我们就可以用此looper来 创建一个Handler,从而实现和它的交互,比如你可以通 ...
- iOS 学习 - 6.Objective-C中的各种遍历(迭代)方式
说明:转自文顶顶 一.使用 for 循环 要遍历字典.数组或者是集合,for 循环是最简单也用的比较多的方法 -(void)iteratorWithFor { //////////处理数组////// ...
- 小波说雨燕 第三季 构建 swift UI 之 UI组件集-视图集(五)Image View视图 学习笔记
留下两个问题:1.后面涉及到的异常不知道原因.2.动态图片到了程序里面就不动了. 然后: 上面是有问题的,下面是没有问题的了. 代码(另外简单写的代码,纠正了那个错误): imp ...
- Javascript中的五种数据类型
Undefined 未定义.只有一个值undefined Null 只有一个值,null Boolean 在javascript中,只要逻辑表达式不返回undefined不返回null ...
- n&(n-1) n&(-n)
n&(n-1) n&(-n) n&(n-1)作用:将n的二进制表示中的最低位为1的改为0,先看一个简单的例子:n = 10100(二进制),则(n-1) = 10011 = ...
- javascript/jquery 常见功能实现(持续更新...)
1. input 只能输入整数数字和字母 $(document).on('keyup','#no',function(){ var val = $.trim($(this).val()); if(va ...
- oracle的增删改查语句
创建一个表: cteate table 表名(列1 类型, 列2 类型);查看表结构 desc表名添加一个字段 alter table 表名 add(列类型);修改字段类型 alter table 表 ...
- 如何在WIN7下进行LINUX虚拟机搭建
Linux是一套免费使用和自由传播的类Unix操作系统,非常适用于搭建网络服务器等,我本人日常工作时,是使用的LINUX和WIN7双操作系统,但每次更换系统总要关机重启很不方便,所以也在WIN7下搭建 ...