非递归中序遍历

出处:http://blog.csdn.net/sgbfblog/article/details/7773103


在前一篇文章二叉树遍历递归算法对二叉树遍历的递归算法做了总结,这篇文章就来对二叉树遍历的非递归算法做个汇总。还是与上一篇文章一样的顺序,一一汇总先序、中序、后序以及层序遍历的非递归算法。


1、先序遍历(非递归算法)

先序遍历非递归访问,使用栈即可实现。先序遍历的非递归访问在所有的遍历中算是最简单的了。主要思想就是先将根结点压入栈,然后根结点出栈并访问根结点,而后依次将根结点的右孩子、左孩子入栈,直到栈为空为止。代码如下:

[cpp]  view plain copy
  1. void preOrderIter(struct node *root)  
  2. {  
  3.     if (root == NULL) return;  
  4.     stack<struct node *> s;  
  5.     s.push(root);  
  6.     while (!s.empty()) {  
  7.         struct node *nd = s.top();  
  8.         cout << nd->data << " ";  
  9.         s.pop();  
  10.         if (nd->right != NULL)  
  11.             s.push(nd->right);  
  12.         if (nd->left != NULL)  
  13.             s.push(nd->left);  
  14.     }  
  15.     cout << endl;  
  16. }  

先序遍历的非递归算法另一算法,也是用的栈,只是稍微复杂点,当左子树遍历完后,需要回溯并遍历右子树。

[cpp]  view plain copy
  1. void preOrderIter2(struct node *root)  
  2. {  
  3.     stack<struct node *> s;  
  4.     while (root != NULL || !s.empty()) {  
  5.         if (root != NULL) {  
  6.             cout << root->data << " "//访问结点并入栈  
  7.             s.push(root);                
  8.             root = root->left;         //访问左子树  
  9.         } else {  
  10.             root = s.top();            //回溯至父亲结点  
  11.             s.pop();  
  12.             root = root->right;        //访问右子树  
  13.         }  
  14.     }  
  15.     cout << endl;  
  16. }  
本算法有一个地方要注意的是,每次从栈中pop出结点时,表示该结点以及该的左子树已经访问完了,接下来访问其右子树。

2、中序遍历(非递归算法)

中序遍历非递归算法也是采用栈实现,与上面的先序遍历算法2类似,只是访问根结点的时机不同。

[cpp]  view plain copy
  1. void inOrderIter(struct node *root)  
  2. {  
  3.     stack<struct node *> s;  
  4.     while (root != NULL || !s.empty()) {  
  5.         if (root != NULL) {  
  6.             s.push(root);  
  7.             root = root->left;  
  8.         }  
  9.         else {  
  10.             root = s.top();  
  11.             cout << root->data << " ";  //访问完左子树后才访问根结点  
  12.             s.pop();  
  13.             root = root->right;        //访问右子树  
  14.         }  
  15.     }  
  16.     cout << endl;  
  17. }  

当然,中序遍历非递归算法还有很多种,比如修改二叉树结点结构加入一个字段来标示结点是否被访问过等,但是都比该算法复杂,且修改了原来的二叉树结构,所以在这里就不啰嗦了,有兴趣的可以移步这里二叉树非递归访问,后续如果有时间我会对二叉树遍历非递归算法通过在结点添加一个标记的方法再做一个总结。


3、后序遍历(非递归算法)

后序遍历的非递归算法较复杂,使用一个栈可以实现,但是过程很繁琐,这里可以巧妙的用两个栈来实现后序遍历的非递归算法。注意到后序遍历可以看作是下面遍历的逆过程:即先遍历某个结点,然后遍历其右孩子,然后遍历其左孩子。这个过程逆过来就是后序遍历。算法步骤如下:

  1. Push根结点到第一个栈s中。
  2. 从第一个栈s中Pop出一个结点,并将其Push到第二个栈output中。
  3. 然后Push结点的左孩子和右孩子到第一个栈s中。
  4. 重复过程2和3直到栈s为空。
  5. 完成后,所有结点已经Push到栈output中,且按照后序遍历的顺序存放,直接全部Pop出来即是二叉树后序遍历结果。

[cpp]  view plain copy
  1. void postOrderIter(struct node *root)  
  2. {  
  3.     if (!root) return;  
  4.     stack<struct node*> s, output;  
  5.     s.push(root);  
  6.     while (!s.empty()) {  
  7.         struct node *curr = s.top();  
  8.         output.push(curr);  
  9.         s.pop();  
  10.         if (curr->left)  
  11.             s.push(curr->left);  
  12.         if (curr->right)  
  13.             s.push(curr->right);  
  14.     }  
  15.       
  16.     while (!output.empty()) {  
  17.         cout << output.top()->data << " ";  
  18.         output.pop();  
  19.     }  
  20.     cout << endl;  
  21. }  

4、层序遍历(非递归算法)

如果不考虑分层换行打印,则用一个队列可以很容易的通过非递归实现层序遍历。但是要每打印一层换一行,就显得稍微复杂了一点。可以有两种方法,第一种使用两个队列,代码很简练,第二种方法则是使用一个队列,代码稍显复杂。

方法一:使用两个队列

第一个队列currentLevel用于存储当前层的结点,第二个队列nextLevel用于存储下一层的结点。当前层currentLevel为空时,表示这一层已经遍历完成,可以打印换行符了。

然后将第一个空的队列currentLevel与队列nextLevel交换,然后重复该过程直到结束。这个算法比较好理解。

[cpp]  view plain copy
  1. void levelOrderIter(struct node* root)  
  2. {  
  3.     if (!root) return;  
  4.     queue<struct node *> currentLevel, nextLevel;  
  5.     currentLevel.push(root);  
  6.     while (!currentLevel.empty()) {  
  7.         struct node *currNode = currentLevel.front();  
  8.         currentLevel.pop();  
  9.         if (currNode) {  
  10.             cout << currNode->data << " ";  
  11.             nextLevel.push(currNode->left);  
  12.             nextLevel.push(currNode->right);  
  13.         }  
  14.         if (currentLevel.empty()) {  
  15.             cout << endl;  
  16.             swap(currentLevel, nextLevel);  
  17.         }  
  18.     }  
  19. }  
  20.   
  21. void swap(queue<struct node *> &curr, queue<struct node*> &next)  
  22. {  
  23.     while (!next.empty()) {  
  24.         struct node *nd = next.front();  
  25.         next.pop();  
  26.         curr.push(nd);  
  27.     }  
  28. }  

方法二:使用一个队列

只使用一个队列的话,需要额外的两个变量来保存当前层结点数目以及下一层的结点数目。

[cpp]  view plain copy
  1. void levelOrderIter2(struct node *root)  
  2. {  
  3.     if (!root) return;  
  4.     queue<struct node*> nodesQueue;  
  5.     int nodesInCurrentLevel = 1;  
  6.     int nodesInNextLevel = 0;  
  7.     nodesQueue.push(root);  
  8.     while (!nodesQueue.empty()) {  
  9.         struct node *currNode = nodesQueue.front();  
  10.         nodesQueue.pop();  
  11.         nodesInCurrentLevel--;  
  12.         if (currNode) {  
  13.             cout << currNode->data << " ";  
  14.             nodesQueue.push(currNode->left);  
  15.             nodesQueue.push(currNode->right);  
  16.             nodesInNextLevel += 2;  
  17.         }  
  18.         if (nodesInCurrentLevel == 0) {  
  19.             cout << endl;  
  20.             nodesInCurrentLevel = nodesInNextLevel;  
  21.             nodesInNextLevel = 0;  
  22.         }  
  23.     }  
  24. }  
1.先序遍历非递归算法#define maxsize 100typedef struct{ Bitree Elem[maxsize]; int top;}SqStack;void PreOrderUnrec(Bitree t){ SqStack s; StackInit(s); p=t; while (p!=null || !StackEmpty(s)) { while (p!=null) //遍历左子树 { visite(p->data); push(s,p); p=p->lchild; }//endwhile if (!StackEmpty(s)) //通过下一次循环中的内嵌while实现右子树遍历 { p=pop(s); p=p->rchild; }//endif }//endwhile }//PreOrderUnrec2.中序遍历非递归算法#define maxsize 100typedef struct{ Bitree Elem[maxsize]; int top;}SqStack;void InOrderUnrec(Bitree t){ SqStack s; StackInit(s); p=t; while (p!=null || !StackEmpty(s)) { while (p!=null) //遍历左子树 { push(s,p); p=p->lchild; }//endwhile if (!StackEmpty(s)) { p=pop(s); visite(p->data); //访问根结点 p=p->rchild; //通过下一次循环实现右子树遍历 }//endif }//endwhile}//InOrderUnrec3.后序遍历非递归算法#define maxsize 100typedef enum{L,R} tagtype;typedef struct { Bitree ptr; tagtype tag;}stacknode;typedef struct{ stacknode Elem[maxsize]; int top;}SqStack;void PostOrderUnrec(Bitree t){ SqStack s; stacknode x; StackInit(s); p=t; do { while (p!=null) //遍历左子树 { x.ptr = p; x.tag = L; //标记为左子树 push(s,x); p=p->lchild; } while (!StackEmpty(s) && s.Elem[s.top].tag==R) { x = pop(s); p = x.ptr; visite(p->data); //tag为R,表示右子树访问完毕,故访问根结点 } if (!StackEmpty(s)) { s.Elem[s.top].tag =R; //遍历右子树 p=s.Elem[s.top].ptr->rchild; } }while (!StackEmpty(s));}//PostOrderUnrec
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值