【經典面試題二】二叉樹的遞歸與非遞歸遍歷(前序、中序、後序)

【寫在前面】

  二叉樹是一種很是重要的數據結構,不少其它數據結構都是基於二叉樹的基礎演變而來的。對於二叉樹,有前序、中序以及後序三種遍歷方法。由於樹的定義自己就 是遞歸定義,所以採用遞歸的方法去實現樹的三種遍歷不只容易理解並且代碼很簡潔。而對於樹的遍歷若採用非遞歸的方法,就要採用棧去模擬實現。在三種遍歷 中,前序和中序遍歷的非遞歸算法都很容易實現,非遞歸後序遍歷實現起來相對來講要難一點。node

一.前序遍歷

   前序遍歷按照「根結點-左孩子-右孩子」的順序進行訪問。ios

   1.遞歸實現

1 void preOrder1(BinTree *root)     //遞歸前序遍歷 
2 {
3     if(root!=NULL)
4     {
5         cout<<root->data<<" ";
6         preOrder1(root->lchild);
7         preOrder1(root->rchild);
8     }
9 }

   2.非遞歸實現

  根據前序遍歷訪問的順序,優先訪問根結點,而後再分別訪問左孩子和右孩子。即對於任一結點,其可看作是根結點,所以能夠直接訪問,訪問完以後,若其左孩子不爲空,按相同規則訪問它的左子樹;當訪問其左子樹時,再訪問它的右子樹。所以其處理過程以下:算法

  對於任一結點P:數據結構

     1)訪問結點P,並將結點P入棧;post

     2)判斷結點P的左孩子是否爲空,若爲空,則取棧頂結點並進行出棧操做,並將棧頂結點的右孩子置爲當前的結點P,循環至1);若不爲空,則將P的左孩子置爲當前的結點P;spa

     3)直到P爲NULL而且棧爲空,則遍歷結束。code

 1 void preOrder2(BinTree *root)     //非遞歸前序遍歷 
 2 {
 3     stack<BinTree*> s;
 4     BinTree *p=root;
 5     while(p!=NULL||!s.empty())
 6     {
 7         while(p!=NULL)
 8         {
 9             cout<<p->data<<" ";
10             s.push(p);
11             p=p->lchild;
12         }
13         if(!s.empty())
14         {
15             p=s.top();
16             s.pop();
17             p=p->rchild;
18         }
19     }
20 }

二.中序遍歷

  中序遍歷按照「左孩子-根結點-右孩子」的順序進行訪問。blog

  1.遞歸實現

1 void inOrder1(BinTree *root)      //遞歸中序遍歷
2 {
3     if(root!=NULL)
4     {
5         inOrder1(root->lchild);
6         cout<<root->data<<" ";
7         inOrder1(root->rchild);
8     }
9 }

  2.非遞歸實現

  根據中序遍歷的順序,對於任一結點,優先訪問其左孩子,而左孩子結點又能夠看作一根結點,而後繼續訪問其左孩子結點,直到遇到左孩子結點爲空的結點才進行訪問,而後按相同的規則訪問其右子樹。所以其處理過程以下:遞歸

  對於任一結點P,字符串

   1)若其左孩子不爲空,則將P入棧並將P的左孩子置爲當前的P,而後對當前結點P再進行相同的處理;

   2)若其左孩子爲空,則取棧頂元素並進行出棧操做,訪問該棧頂結點,而後將當前的P置爲棧頂結點的右孩子;

   3)直到P爲NULL而且棧爲空則遍歷結束。

 1 void inOrder2(BinTree *root)      //非遞歸中序遍歷
 2 {
 3     stack<BinTree*> s;
 4     BinTree *p=root;
 5     while(p!=NULL||!s.empty())
 6     {
 7         while(p!=NULL)
 8         {
 9             s.push(p);
10             p=p->lchild;
11         }
12         if(!s.empty())
13         {
14             p=s.top();
15             cout<<p->data<<" ";
16             s.pop();
17             p=p->rchild;
18         }
19     }    
20 } 

三.後序遍歷

       後序遍歷按照「左孩子-右孩子-根結點」的順序進行訪問。

       1.遞歸實現

1 void postOrder1(BinTree *root)    //遞歸後序遍歷
2 {
3     if(root!=NULL)
4     {
5         postOrder1(root->lchild);
6         postOrder1(root->rchild);
7         cout<<root->data<<" ";
8     }    
9 }

  2.非遞歸實現

  後序遍歷的非遞歸實現是三種遍歷方式中最難的一種。由於在後序遍歷中,要保證左孩子和右孩子都已被訪問而且左孩子在右孩子前訪問才能訪問根結點,這就爲流程的控制帶來了難題。下面介紹兩種思路。

      第一種思路:對於任一結點P,將其入棧,而後沿其左子樹一直往下搜索,直到搜索到沒有左孩子的結點,此時該結點出如今棧頂,可是此時不能將其出棧並訪問, 所以其右孩子還爲被訪問。因此接下來按照相同的規則對其右子樹進行相同的處理,當訪問完其右孩子時,該結點又出如今棧頂,此時能夠將其出棧並訪問。這樣就 保證了正確的訪問順序。能夠看出,在這個過程當中,每一個結點都兩次出如今棧頂,只有在第二次出如今棧頂時,才能訪問它。所以須要多設置一個變量標識該結點是 否是第一次出如今棧頂。

 1 void postOrder2(BinTree *root)    //非遞歸後序遍歷
 2 {
 3     stack<BTNode*> s;
 4     BinTree *p=root;
 5     BTNode *temp;
 6     while(p!=NULL||!s.empty())
 7     {
 8         while(p!=NULL)              //沿左子樹一直往下搜索,直至出現沒有左子樹的結點 
 9         {
10             BTNode *btn=(BTNode *)malloc(sizeof(BTNode));
11             btn->btnode=p;
12             btn->isFirst=true;
13             s.push(btn);
14             p=p->lchild;
15         }
16         if(!s.empty())
17         {
18             temp=s.top();
19             s.pop();
20             if(temp->isFirst==true)     //表示是第一次出如今棧頂 
21              {
22                 temp->isFirst=false;
23                 s.push(temp);
24                 p=temp->btnode->rchild;    
25             }
26             else                        //第二次出如今棧頂 
27              {
28                 cout<<temp->btnode->data<<" ";
29                 p=NULL;
30             }
31         }
32     }    
33 }

  第二種思路:要保證根結點在左孩子和右孩子訪問以後才能訪問,所以對於任一結點P,先將其入棧。若是P不存在左孩子和右孩子,則能夠直接訪問它;或者P存 在左孩子或者右孩子,可是其左孩子和右孩子都已被訪問過了,則一樣能夠直接訪問該結點。若非上述兩種狀況,則將P的右孩子和左孩子依次入棧,這樣就保證了 每次取棧頂元素的時候,左孩子在右孩子前面被訪問,左孩子和右孩子都在根結點前面被訪問。

 1 void postOrder3(BinTree *root)     //非遞歸後序遍歷
 2 {
 3     stack<BinTree*> s;
 4     BinTree *cur;                      //當前結點 
 5     BinTree *pre=NULL;                 //前一次訪問的結點 
 6     s.push(root);
 7     while(!s.empty())
 8     {
 9         cur=s.top();
10         if((cur->lchild==NULL&&cur->rchild==NULL)||
11            (pre!=NULL&&(pre==cur->lchild||pre==cur->rchild)))
12         {
13             cout<<cur->data<<" ";  //若是當前結點沒有孩子結點或者孩子節點都已被訪問過 
14               s.pop();
15             pre=cur; 
16         }
17         else
18         {
19             if(cur->rchild!=NULL)
20                 s.push(cur->rchild);
21             if(cur->lchild!=NULL)    
22                 s.push(cur->lchild);
23         }
24     }    
25 }

四.整個程序完整的代碼

  1 /*二叉樹的遍歷* 2011.8.25*/ 
  2 
  3 #include <iostream>
  4 #include<string.h>
  5 #include<stack> 
  6 using namespace std;
  7 
  8 typedef struct node
  9 {
 10     char data;
 11     struct node *lchild,*rchild;
 12 }BinTree;
 13 
 14 typedef struct node1
 15 {
 16     BinTree *btnode;
 17     bool isFirst;
 18 }BTNode;
 19 
 20 
 21 void creatBinTree(char *s,BinTree *&root)  //建立二叉樹,s爲形如A(B,C(D,E))形式的字符串 
 22 {
 23     int i;
 24     bool isRight=false;
 25     stack<BinTree*> s1;          //存放結點 
 26     stack<char> s2;              //存放分隔符
 27     BinTree *p,*temp;
 28     root->data=s[0];
 29     root->lchild=NULL;
 30     root->rchild=NULL;
 31     s1.push(root);
 32     i=1;
 33     while(i<strlen(s))
 34     {
 35         if(s[i]=='(')
 36         {
 37             s2.push(s[i]);
 38             isRight=false;
 39         }    
 40         else if(s[i]==',')    
 41         {
 42             isRight=true;
 43         }
 44         else if(s[i]==')')
 45         {
 46             s1.pop();
 47             s2.pop();
 48         }
 49         else if(isalpha(s[i]))
 50         {
 51             p=(BinTree *)malloc(sizeof(BinTree));
 52             p->data=s[i];
 53             p->lchild=NULL;
 54             p->rchild=NULL;
 55             temp=s1.top();
 56             if(isRight==true)    
 57             {
 58                 temp->rchild=p;
 59                 cout<<temp->data<<"的右孩子是"<<s[i]<<endl;
 60             }
 61             else
 62             {
 63                 temp->lchild=p;
 64                 cout<<temp->data<<"的左孩子是"<<s[i]<<endl;
 65             }
 66             if(s[i+1]=='(')
 67                 s1.push(p);
 68         }
 69         i++;
 70     }    
 71 }
 72 
 73 void display(BinTree *root)        //顯示樹形結構 
 74 {
 75     if(root!=NULL)
 76     {
 77         cout<<root->data;
 78         if(root->lchild!=NULL)
 79         {
 80             cout<<'(';
 81             display(root->lchild);
 82         }
 83         if(root->rchild!=NULL)
 84         {
 85             cout<<',';
 86             display(root->rchild);
 87             cout<<')';
 88         }
 89     }
 90 }
 91 
 92 void preOrder1(BinTree *root)     //遞歸前序遍歷 
 93 {
 94     if(root!=NULL)
 95     {
 96         cout<<root->data<<" ";
 97         preOrder1(root->lchild);
 98         preOrder1(root->rchild);
 99     }
100 }
101 
102 void inOrder1(BinTree *root)      //遞歸中序遍歷
103 {
104     if(root!=NULL)
105     {
106         inOrder1(root->lchild);
107         cout<<root->data<<" ";
108         inOrder1(root->rchild);
109     }
110 } 
111 
112 void postOrder1(BinTree *root)    //遞歸後序遍歷
113 {
114     if(root!=NULL)
115     {
116         postOrder1(root->lchild);
117         postOrder1(root->rchild);
118         cout<<root->data<<" ";
119     }    
120 } 
121 
122 void preOrder2(BinTree *root)     //非遞歸前序遍歷 
123 {
124     stack<BinTree*> s;
125     BinTree *p=root;
126     while(p!=NULL||!s.empty())
127     {
128         while(p!=NULL)
129         {
130             cout<<p->data<<" ";
131             s.push(p);
132             p=p->lchild;
133         }
134         if(!s.empty())
135         {
136             p=s.top();
137             s.pop();
138             p=p->rchild;
139         }
140     }
141 }
142 
143 void inOrder2(BinTree *root)      //非遞歸中序遍歷
144 {
145     stack<BinTree*> s;
146     BinTree *p=root;
147     while(p!=NULL||!s.empty())
148     {
149         while(p!=NULL)
150         {
151             s.push(p);
152             p=p->lchild;
153         }
154         if(!s.empty())
155         {
156             p=s.top();
157             cout<<p->data<<" ";
158             s.pop();
159             p=p->rchild;
160         }
161     }    
162 } 
163 
164 void postOrder2(BinTree *root)    //非遞歸後序遍歷
165 {
166     stack<BTNode*> s;
167     BinTree *p=root;
168     BTNode *temp;
169     while(p!=NULL||!s.empty())
170     {
171         while(p!=NULL)              //沿左子樹一直往下搜索,直至出現沒有左子樹的結點 
172          {
173             BTNode *btn=(BTNode *)malloc(sizeof(BTNode));
174             btn->btnode=p;
175             btn->isFirst=true;
176             s.push(btn);
177             p=p->lchild;
178         }
179         if(!s.empty())
180         {
181             temp=s.top();
182             s.pop();
183             if(temp->isFirst==true)     //表示是第一次出如今棧頂 
184              {
185                 temp->isFirst=false;
186                 s.push(temp);
187                 p=temp->btnode->rchild;    
188             }
189             else                        //第二次出如今棧頂 
190              {
191                 cout<<temp->btnode->data<<" ";
192                 p=NULL;
193             }
194         }
195     }    
196 } 
197 
198 void postOrder3(BinTree *root)     //非遞歸後序遍歷
199 {
200     stack<BinTree*> s;
201     BinTree *cur;                      //當前結點 
202     BinTree *pre=NULL;                 //前一次訪問的結點 
203     s.push(root);
204     while(!s.empty())
205     {
206         cur=s.top();
207         if((cur->lchild==NULL&&cur->rchild==NULL)||
208            (pre!=NULL&&(pre==cur->lchild||pre==cur->rchild)))
209         {
210             cout<<cur->data<<" ";  //若是當前結點沒有孩子結點或者孩子節點都已被訪問過 
211               s.pop();
212             pre=cur; 
213         }
214         else
215         {
216             if(cur->rchild!=NULL)
217                 s.push(cur->rchild);
218             if(cur->lchild!=NULL)    
219                 s.push(cur->lchild);
220         }
221     }    
222 }
223 
224 
225 int main(int argc, char *argv[])
226 {
227     char s[100];
228     while(scanf("%s",s)==1)
229     {
230         BinTree *root=(BinTree *)malloc(sizeof(BinTree));
231         creatBinTree(s,root);
232         display(root);
233         cout<<endl;
234         preOrder2(root);
235         cout<<endl; 
236         inOrder2(root);
237         cout<<endl;
238         postOrder2(root);
239         cout<<endl;
240         postOrder3(root);
241         cout<<endl;
242     }
243     return 0;
244 }

 

 

做者:海子

本博客中未標明轉載的文章歸做者海子和博客園共有,歡迎轉載,但未經做者贊成必須保留此段聲明,且在文章頁面明顯位置給出原文鏈接,不然保留追究法律責任的權利。  

相關文章
相關標籤/搜索