NYOJ--重建二叉樹
重建二叉樹
- 描述
- 題目很簡單,給你一棵二叉樹的後序和中序序列,求出它的前序序列(So easy!)。
- 輸入
-
輸入有多組資料(少於100組),以檔案結尾結束。
每組資料僅一行,包括兩個字串,中間用空格隔開,分別表示二叉樹的後序和中序序列(字串長度小於26,輸入資料保證合法)。 - 輸出
- 每組輸出資料單獨佔一行,輸出對應得先序序列。
- 樣例輸入
-
ACBFGED ABCDEFG CDAB CBAD
- 樣例輸出
-
DBACEGF BCAD
- 來源
解析:這是一道資料結構中很水的題,可惜自己因為檔案結束符EOF這個TLE了很多次,唉……粗心哈!先簡單說一下二叉樹的遍歷
先序遍歷二叉樹:先序遍歷根節點,先序遍歷左子樹,先序遍歷右子樹,也就是NLR
中序遍歷二叉樹:中序遍歷左子樹,中序遍歷根節點,中序遍歷右子樹,也就是LNR
後序遍歷二叉樹:後序遍歷左子樹,後序遍歷右子樹,後序遍歷根節點,也就是LRN
要想由這三種遍歷二叉樹的序列重建唯一的二叉樹則至少得有兩種遍歷序列,且這兩種中一定要有中序序列,也就是先序+中序-->二叉樹 或者 後序+中序-->二叉樹,而先序+中序得不到唯一的二叉樹。
下面看一下這三種遍歷的特點,首先先序序列總是先遍歷根節點,所以它的序列的第一個必然是整個二叉樹的根節點,相反,後序遍歷的序列最後一個必然是二叉樹的根節點,通過缺點了二叉樹的根節點,在中序序列中找到這個根節點,因為中序序列是先遍歷左字數再遍歷根節點,然後是右子樹,所以在中序序列中的根節點左邊的節點全都是左子樹的,右邊的序列都應該是右子樹的,接著遞迴的應用就可以得到整個二叉樹了(不知道說清楚的沒)
後序序列:ACBFGED
中序序列:ABCDEFG
首先確定D是整個二叉樹的根節點,然後在中序序列中找到D則其左邊的ABC就是D的左子樹,EFG就是 D的右子樹。
提取出左字數的後序和中序序列
後序序列:ACB
中序序列: ABC
後序序列的最後一個字元是B則B是當前子樹的根節點,然後在中序序列中找到B則左邊的A是B的左子樹,C是B的右子樹。
以此遞推就可以得到整個二叉樹的結構了,如下圖所示
同樣知道先序序列和中序序列後也可以構建出唯一的二叉樹,掌握的思想,下面就是用程式碼實現的事情了。
程式碼本著重建二叉樹的原則,並沒考慮太多OJ時間和空間的限制,但AC還是可以的,在程式碼中我右加上了先序+中序-->二叉樹的函式
當然在實際的OJ上做類似的題時不用那麼麻煩根根節點分配空間,直接輸出就好了,我只不過是想將二叉樹建立起來,看起來比較直觀而已
#include <stdio.h> #include <malloc.h> #include <string.h> //二叉連結串列 typedef struct node{ char data;//節點資料元素 struct node *lchild;//指向左孩子 struct node *rchild;//指向右孩子 }BiNode,*BTree; //利用後序和中序建立二叉樹 void GetPreOrder(char *last,char *mid,BTree &T,int len) { if(len==0) { T = NULL; return; } //取出後序序列中的最後一個節點 char ch=last[len-1]; int index=0; //在中序序列中進行查詢根節點,並用index記錄其在序列中的索引 while(mid[index]!=ch) { index++; } //給根節點分配空間 T=(BTree)malloc(sizeof(BiNode)); T->data=mid[index]; //建立左子樹 GetPreOrder(last,mid,T->lchild,index); //建立右子樹 GetPreOrder(last+index,mid+index+1,T->rchild,len-index-1); } void GetPostOrder(char *prim,char *mid,BTree &T,int len) { if(len==0) { T=NULL; return; } //提出先序序列中的第一個節點 char ch=prim[0]; int index=0; //在中序序列中查詢當前根節點,並用index記錄其在序列中的位置 while(mid[index]!=ch) { index++; } //給根節點分配空間 T=(BTree)malloc(sizeof(BiNode)); T->data=mid[index]; //建立左子樹 GetPostOrder(prim+1,mid,T->lchild,index); //建立右子樹 GetPostOrder(prim+index+1,mid+index+1,T->rchild,len-index-1); } //先序輸出二叉樹 void PreOrder(BTree T) { if(T!=NULL) { printf("%c",T->data); PreOrder(T->lchild); PreOrder(T->rchild); } } //後序輸出二叉樹 void PostOrder(BTree T) { if(T!=NULL) { PostOrder(T->lchild); PostOrder(T->rchild); printf("%c",T->data); } } int main() { char first[26],mid[26],last[26]; while(scanf("%s%s",last,mid)!=EOF) { BTree T=NULL; GetPreOrder(last,mid,T,strlen(last)); PreOrder(T); // GetPostOrder(last,mid,T,strlen(last)); // PostOrder(T); printf("\n"); } return 0; }
相關文章
- 重建二叉樹二叉樹
- 重建二叉樹[by Python]二叉樹Python
- 面試題7:重建二叉樹面試題二叉樹
- JZ-004-重建二叉樹二叉樹
- 劍指offer——重建二叉樹二叉樹
- 二叉樹的子結構、深度以及重建二叉樹二叉樹
- 劍指offer(四)重建二叉樹二叉樹
- 劍指 Offer 07. 重建二叉樹二叉樹
- 滿二叉樹、完全二叉樹、平衡二叉樹、二叉搜尋樹(二叉查詢樹)和最優二叉樹二叉樹
- 根據前序遍歷序列、中序遍歷序列,重建二叉樹二叉樹
- 二叉樹 & 二叉查詢樹二叉樹
- 劍指offer(java實現)第4題“重建二叉樹”-牛客網Java二叉樹
- 【劍指offer】【4】根據前序和中序結果,重建二叉樹二叉樹
- 排序二叉樹和平衡二叉樹排序二叉樹
- 二叉查詢樹(二叉排序樹)排序
- 二叉樹(順序儲存二叉樹,線索化二叉樹)二叉樹
- 劍指offer:輸入某二叉樹的前序遍歷和中序遍歷的結果,請重建出該二叉樹。二叉樹
- 手擼二叉樹——二叉查詢樹二叉樹
- 手擼二叉樹——AVL平衡二叉樹二叉樹
- 資料結構之樹結構概述(含滿二叉樹、完全二叉樹、平衡二叉樹、二叉搜尋樹、紅黑樹、B-樹、B+樹、B*樹)資料結構二叉樹
- 二叉樹二叉樹
- 二叉樹的應用(1)--二叉樹排序樹基本操作二叉樹排序
- 判斷二叉樹是否為滿二叉樹二叉樹
- 資料結構中的樹(二叉樹、二叉搜尋樹、AVL樹)資料結構二叉樹
- NYOJ--字首式計算
- 自己動手作圖深入理解二叉樹、滿二叉樹及完全二叉樹二叉樹
- 二叉樹、B樹以及B+樹二叉樹
- 平衡二叉樹,B樹,B+樹二叉樹
- 深入學習二叉樹 (一) 二叉樹基礎二叉樹
- 相同二叉樹和鏡面二叉樹問題二叉樹
- 判斷某棵二叉樹是否二叉排序樹二叉樹排序
- 平衡二叉樹(AVL樹)和 二叉排序樹轉化為平衡二叉樹 及C語言實現二叉樹排序C語言
- 樹和二叉樹簡介二叉樹
- n叉樹vs二叉樹二叉樹
- Chapter 3 樹與二叉樹APT二叉樹
- 樹(2)--二叉樹的遍歷(非遞迴)+線索二叉樹二叉樹遞迴
- 迭代二叉樹二叉樹
- 二叉樹深度二叉樹