數據結構】后綴表達式-->表達式樹


原文,轉載如下:

用到了棧,並且遞歸實現了中序遍歷,后序遍歷,前序遍歷。

同時應該學會union的使用方法。

 

基礎知識:

一、表達式樹
    
    表達式樹的樹葉是操作數(operand),加常數或變量名字,而其他的結點為操作數(operator)。由於這里所有的操作都是二元的,因此這棵特定的樹正好是二叉樹,雖然這是最簡單的情況,但是結點還是有可能含有多於兩個的兒子,這里我們不討論。  
 
 
二、構造一棵表達式樹
 
    之前我們實現過一個中綴表達式求值的具體程序,在求值過程中需要用兩個棧,並且代碼並不簡單。而這里你會看到,對於表達式樹的求值操作卻非常簡單,甚至只需要兩條語句。因為這里大部分操作都是遞歸定義,二遞歸函數本身都是很簡潔的,甚至比你想象的還要簡單,甚至只需要兩條語句。因為這里大部分操作都是遞歸定義,二遞歸函數本身都是很簡潔的,甚至比你想象的還要簡單!就像樹的遍歷操作。三種遍歷分別是先序遍歷、中序遍歷與后序遍歷,正好對應表達式的三種形式:前綴型、中綴型與后綴型。其中為大家熟知的是中綴形式,如2+3*(5-4)。前綴型表達式又叫波蘭式(Polish Notation),后綴性表達式又叫逆波蘭式(Reverse Polish Notation)。他們最早於1920年波蘭數學家Jan Lukasiewicz發明,這兩種表示方式的最大特點是不需要括號來表明優先級,他們經常用於計算機科學,特別是編譯器設計方面。
 
    下面給出一種算法來把后綴表達式轉變成表達式樹。我們一次一個符號地讀入表達式。如果符號是操作數,那么就建立一個單結點樹並將它推入棧中。如果符號是操作符,那么就從棧中彈出兩棵樹T1和T2(T1先彈出)並形成一棵新的樹,該樹的根就是操作符,它的左、右兒子分別是T2和T1。然后將指向這顆樹的指針壓入棧中。
 
    下面來看一個例子。設輸入為ab+cde+**
 
前兩個符號是操作數,因此創建兩棵單結點樹並將指向它們的指針壓入棧中。
 
 
接着,"+"被讀入,因此指向兩棵樹的指針被彈出,形成一棵新的樹,並將指向它的指針壓入棧中。
 

然后,c,d和e被讀入,在單個結點樹創建后,指向對應的樹的指針被壓入棧中。
 
 
接下來讀入"+"號,因此兩棵樹合並。
 
 
繼續進行,讀入"*"號,因此,彈出兩棵樹的指針合並形成一棵新的樹,"*"號是它的根。
 
 
最后,讀入一個符號,兩棵樹合並,而指向最后的樹的指針被留在棧中。
 
 

 

  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3.   
  4. #define MAX 100  
  5.   
  6. /* 樹結點的設計 */  
  7. typedef struct node  
  8. {  
  9.     /* 數字和運算符 */  
  10.     union  
  11.     {  
  12.         char operator;  
  13.         int data;  
  14.     };  
  15.   
  16.     struct node *lchild;  
  17.     struct node *rchild;  
  18. }TreeNode;  
  19.   
  20. /* 樹棧 */  
  21. typedef struct Tree_Stack  
  22. {  
  23.     TreeNode *buf[MAX];  
  24.     int n;  
  25. }TreeStack;  
  26.   
  27. /* 創建空棧 */  
  28. TreeStack *create_empty_stack()  
  29. {  
  30.     TreeStack *pstack;  
  31.   
  32.     pstack = (TreeStack *)malloc(sizeof(TreeStack));  
  33.     pstack->n = -1;  
  34.   
  35.     return pstack;  
  36. }  
  37.   
  38. /* 入棧 */  
  39. int push_stack(TreeStack *p, TreeNode *data)  
  40. {  
  41.     p->n++;  
  42.     p->buf[p->n] = data;  
  43.   
  44.     return 0;  
  45. }  
  46.   
  47. /* 出棧 */  
  48. TreeNode *pop_stack(TreeStack *p)  
  49. {  
  50.     TreeNode *data;  
  51.   
  52.     data = p->buf[p->n];  
  53.     p->n --;  
  54.   
  55.     return data;  
  56. }  
  57.   
  58. /* 判斷棧空 */  
  59. int is_empty_stack(TreeStack *p)  
  60. {  
  61.     if(p->n == -1)  
  62.         return 1;  
  63.     else  
  64.         return 0;  
  65. }  
  66.   
  67. /* 創建后綴表達式樹 */  
  68. TreeNode *create_express_tree(char *str, TreeStack *p)  
  69. {  
  70.     int i = 0;  
  71.     TreeNode *current;  
  72.     TreeNode *left, *right;  
  73.   
  74.     while(str[i])  
  75.     {  
  76.         if(str[i] >= '0' && str[i] <= '9')  
  77.         {  
  78.             current = (TreeNode *)malloc(sizeof(TreeNode));  
  79.             current->data = str[i] - '0';  
  80.             current->lchild = NULL;  
  81.             current->rchild = NULL;  
  82.             push_stack(p, current);  
  83.         }  
  84.         else  
  85.         {  
  86.             current = (TreeNode *)malloc(sizeof(TreeNode));  
  87.             current->operator = str[i];  
  88.             right = pop_stack(p);  
  89.             left = pop_stack(p);  
  90.             current->lchild = left;  
  91.             current->rchild = right;  
  92.             push_stack(p, current);  
  93.         }  
  94.         i++;  
  95.     }  
  96.     return p->buf[p->n];  
  97. }  
  98.   
  99. /* 打印結點 */  
  100. void print_node(TreeNode *p)  
  101. {  
  102.     if(p->lchild == NULL && p->rchild == NULL)  
  103.         printf("%d ", p->data);  
  104.     else  
  105.         printf("%c ", p->operator);  
  106. }  
  107.   
  108. /* 訪問結點 */  
  109. int visit_node(TreeNode *p)  
  110. {  
  111.     print_node(p);  
  112.   
  113.     return 0;  
  114. }  
  115.   
  116. /* 樹的后序遍歷 */  
  117. void PostOrder(TreeNode *p)  
  118. {  
  119.     if(p != NULL)  
  120.     {  
  121.         PostOrder(p->lchild);  
  122.         PostOrder(p->rchild);  
  123.         visit_node(p);  
  124.     }  
  125. }  
  126.   
  127. /* 樹的中序遍歷 */  
  128. void InOrder(TreeNode *p)  
  129. {  
  130.     if(p != NULL)  
  131.     {  
  132.         InOrder(p->lchild);  
  133.         visit_node(p);  
  134.         InOrder(p->rchild);  
  135.     }  
  136. }  
  137.   
  138. /* 樹的前序遍歷 */  
  139. void PreOrder(TreeNode *p)  
  140. {  
  141.     if(p != NULL)  
  142.     {  
  143.         visit_node(p);  
  144.         PreOrder(p->lchild);  
  145.         PreOrder(p->rchild);  
  146.     }  
  147. }  
  148.   
  149. int main()  
  150. {  
  151.     TreeNode *thead;  
  152.     TreeStack *pstack;  
  153.     int i = 0;  
  154.     char buf[100];  
  155.   
  156.     scanf("%s", buf);  
  157.   
  158.     pstack = create_empty_stack();  
  159.     thead = create_express_tree(buf, pstack);  
  160.   
  161.     printf("PostOrder:  ");  
  162.     PostOrder(thead);  
  163.     printf("\n");  
  164.   
  165.     printf("InOrder:  ");  
  166.     InOrder(thead);  
  167.     printf("\n");  
  168.   
  169.     printf("PreOrder:  ");  
  170.     PreOrder(thead);  
  171.     printf("\n");  
  172.   
  173.     return 0;  
  174. }  

測試結果如下:


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM