本篇爲關於樹的編程題,給出編譯器 C++(g++)
的解答。主要記錄題意理解和代碼學習過程。node
給定兩棵樹T1和T2。若是T1能夠經過若干次左右孩子互換就變成T2,則咱們稱兩棵樹是「同構」的。例如圖1給出的兩棵樹就是同構的,由於咱們把其中一棵樹的結點A、B、G的左右孩子互換後,就獲得另一棵樹。而圖2就不是同構的。ios
現給定兩棵樹,請你判斷它們是不是同構的。編程
輸入格式:
輸入給出2棵二叉樹樹的信息。對於每棵樹,首先在一行中給出一個非負整數N (≤10),即該樹的結點數(此時假設結點從0到N−1編號);隨後N行,第i行對應編號第i個結點,給出該結點中存儲的1個英文大寫字母、其左孩子結點的編號、右孩子結點的編號。若是孩子結點爲空,則在相應位置上給出「-」。給出的數據間用一個空格分隔。注意:題目保證每一個結點中存儲的字母是不一樣的。函數
輸出格式:
若是兩棵樹是同構的,輸出「Yes」,不然輸出「No」。學習
輸入樣例1(對應圖1):測試
8 A 1 2 B 3 4 C 5 - D - - E 6 - G 7 - F - - H - - 8 G - 4 B 7 6 F - - A 5 1 H - - C 0 - D - - E 2 -
輸出樣例1:ui
Yes
輸入樣例2(對應圖2):this
8 B 5 7 F - - A 0 3 C 6 - H - - D - - G 4 - E 1 - 8 D 6 - B 5 - E - - H - - C 0 2 G - 3 F - - A 1 4
輸出樣例2:spa
No
首先理解同構的意思,題目中的同構是指左右孩子相同便可。一種簡單的判斷兩棵樹是否同構的方法是看結點的兒子,若是都同樣,則是同構。很明顯,圖1中的兩棵樹是同構的,而圖2中的兩棵樹C下面的兒子就不一樣,所以它們不一樣構。.net
本題要求咱們輸入兩棵樹的信息,判斷它們是否同構。這棵二叉樹的信息表示如輸入樣例所示,第一個數是整數,告訴咱們這棵樹有幾個結點,對每一個結點來講有三個信息:結點自己,左兒子,右兒子。左右兒子經過編號來表示,若爲空則用-
來表示。但要注意的是這裏沒有規定必定要從根節點來開始編號,即能以任意的順序進行編號。因此要解這道題咱們還須要進行判別根結點在哪裏。
咱們須要的事情有三個:二叉樹表示,建二叉樹,同構判別。
#include <iostream> #include <vector> using namespace std; #define Max_Node 11 #define END -1 typedef struct node { char value; int left; int right; }Node; //獲取樹的輸入,並將輸入的字符合理轉化成整型數字 void CreateTree(vector<Node>& Tree,int N) { char value,left,right; for (int i=0; i<N; ++i) { cin>>value>>left>>right; Tree[i].value=value; if (left=='-') { Tree[i].left=END; }else { Tree[i].left=left-'0'; } if (right=='-') { Tree[i].right=END; }else { Tree[i].right=right-'0'; } } } //尋找樹的樹根:樹根沒有其它的結點指向它 int FindTreeRoot(vector<Node>& Tree,int N) { int Flag[Max_Node]; for (int i=0; i<N; ++i) { Flag[i]=0; } for (int i=0; i<N; ++i) { if (Tree[i].left!=END) { Flag[Tree[i].left]=1; } if (Tree[i].right!=END) { Flag[Tree[i].right]=1; } } int k; for (k=0; k<N; ++k) { if (Flag[k]==0) { break; } } return k; } //遞歸判斷兩樹是否同構 bool IsOmorphic(int Root1,int Root2,vector<Node>& Tree1,vector<Node>& Tree2) { if (Tree1[Root1].value==Tree2[Root2].value) { //兩結點相等,並都是葉子結點 if (Tree1[Root1].left==END && Tree1[Root1].right==END && Tree2[Root2].left==END && Tree2[Root2].right==END) { return true; } //如下四種狀況:兩個結點都是有一個孩子爲空,另外一個子樹不空且這兩個孩子相等的情形 if (Tree1[Tree1[Root1].left].value==Tree2[Tree2[Root2].left].value && Tree1[Root1].right==END && Tree2[Root2].right==END) { return IsOmorphic(Tree1[Root1].left, Tree2[Root2].left, Tree1, Tree2); } if (Tree1[Tree1[Root1].left].value==Tree2[Tree2[Root2].right].value && Tree1[Root1].right==END && Tree2[Root2].left==END) { return IsOmorphic(Tree1[Root1].left, Tree2[Root2].right, Tree1, Tree2); } if (Tree1[Tree1[Root1].right].value==Tree2[Tree2[Root2].left].value && Tree1[Root1].left==END && Tree2[Root2].right==END) { return IsOmorphic(Tree1[Root1].right, Tree2[Root2].left, Tree1, Tree2); } if (Tree1[Tree1[Root1].right].value==Tree2[Tree2[Root2].right].value && Tree1[Root1].left==END && Tree2[Root2].left==END) { return IsOmorphic(Tree1[Root1].right, Tree2[Root2].right, Tree1, Tree2); } //如下兩種:兩個結點的孩子都相等的情形 if (Tree1[Tree1[Root1].left].value==Tree2[Tree2[Root2].left].value && Tree1[Tree1[Root1].right].value==Tree2[Tree2[Root2].right].value) { return (IsOmorphic(Tree1[Root1].left, Tree2[Root2].left, Tree1, Tree2))&&(IsOmorphic(Tree1[Root1].right, Tree2[Root2].right, Tree1, Tree2)); } if (Tree1[Tree1[Root1].left].value==Tree2[Tree2[Root2].right].value && Tree1[Tree1[Root1].right].value==Tree2[Tree2[Root2].left].value) { return (IsOmorphic(Tree1[Root1].left, Tree2[Root2].right, Tree1, Tree2))&&(IsOmorphic(Tree1[Root1].right, Tree2[Root2].left, Tree1, Tree2)); } } //不符合以上7種狀況代表這兩棵樹不一樣構 return false; } int main(int argc, const char * argv[]) { //輸入兩顆二叉樹的信息 int N1=0; cin>>N1; vector<Node> Tree1(Max_Node); CreateTree(Tree1,N1); int N2=0; cin>>N2; vector<Node> Tree2(Max_Node); CreateTree(Tree2,N2); if (N1!=N2) { cout<<"No"; }else { if (N1==0) { cout<<"Yes"; }else { //建二叉樹 int root1=FindTreeRoot(Tree1,N1); int root2=FindTreeRoot(Tree2,N2); //判斷是否同構 if (IsOmorphic(root1, root2, Tree1, Tree2)) { cout<<"Yes"; }else { cout<<"No"; } } } return 0; }
Given a tree, you are supposed to list all the leaves in the order of top down, and left to right.
Input Specification:
Each input file contains one test case. For each case, the first line gives a positive integer N (≤10) which is the total number of nodes in the tree -- and hence the nodes are numbered from 0 to N−1. Then N lines follow, each corresponds to a node, and gives the indices of the left and right children of the node. If the child does not exist, a "-" will be put at the position. Any pair of children are separated by a space.
Output Specification:
For each test case, print in one line all the leaves' indices in the order of top down, and left to right. There must be exactly one space between any adjacent numbers, and no extra space at the end of the line.
Sample Input:
8 1 - - - 0 - 2 7 - - - - 5 - 4 6
Sample Output:
4 1 5
#include <stdio.h> #include <stdlib.h> #include <string.h> int flag=0;//用於判斷結果輸出格式的 struct NodeInf{//樹的節點信息,左右兒子下標 int LeftIndex; int RightIndex; }; struct BinTreeNode{//樹節點 int Element;//編號 struct BinTreeNode* Left; struct BinTreeNode* Right; }; int FindTreeHead(int book[],int n)//查找樹根 { for(int i=0;i<n;i++) if(book[i]==0) return i; } void CreBinTreeAndPriLeaves(int treehead,struct NodeInf nodeinf[])//層序建立樹,同時輸出葉子 { struct BinTreeNode* BinTree;//樹 struct BinTreeNode* Temp; struct BinTreeNode* Queue[15]; int head=0,tail=0; BinTree=(struct BinTreeNode*)malloc(sizeof(struct BinTreeNode)); BinTree->Element=treehead; Queue[tail++]=BinTree; while(head<tail){ if(nodeinf[Queue[head]->Element].LeftIndex!=-1){ Temp=(struct BinTreeNode*)malloc(sizeof(struct BinTreeNode)); Temp->Element=nodeinf[Queue[head]->Element].LeftIndex; Queue[head]->Left=Temp; Queue[tail++]=Temp; } else{ Queue[head]->Left=NULL; } if(nodeinf[Queue[head]->Element].RightIndex!=-1){ Temp=(struct BinTreeNode*)malloc(sizeof(struct BinTreeNode)); Temp->Element=nodeinf[Queue[head]->Element].RightIndex; Queue[head]->Right=Temp; Queue[tail++]=Temp; } else{ Queue[head]->Right=NULL; } if(Queue[head]->Left==NULL&&Queue[head]->Right==NULL){//判斷是否爲葉子 if(flag) printf("%c",' '); printf("%d",Queue[head]->Element); flag=1; } head++; } putchar('\n'); return; } int main() { int n; char ch; struct NodeInf nodeinf[10];//存儲節點信息 int treehead;//樹根 int book[10];//標記是別人兒子的節點,則未標記的就爲樹根 memset(book,0,sizeof(book)); scanf("%d",&n); for(int i=0;i<n;i++){//題目的節點信息是按照節點編號來的,即0到n-1,題目沒說出來...... getchar(); scanf("%c",&ch); if(ch-'0'>=0&&ch-'0'<=9){ nodeinf[i].LeftIndex=ch-'0'; book[ch-'0']=1; } else nodeinf[i].LeftIndex=-1; getchar(); scanf("%c",&ch); if(ch-'0'>=0&&ch-'0'<=9){ nodeinf[i].RightIndex=ch-'0'; book[ch-'0']=1; } else nodeinf[i].RightIndex=-1; } treehead=FindTreeHead(book,n);//找樹根 CreBinTreeAndPriLeaves(treehead,nodeinf); return 0; }
給定一個插入序列就能夠惟一肯定一棵二叉搜索樹。然而,一棵給定的二叉搜索樹卻能夠由多種不一樣的插入序列獲得。例如分別按照序列{2, 1, 3}和{2, 3, 1}插入初始爲空的二叉搜索樹,都獲得同樣的結果。因而對於輸入的各類插入序列,你須要判斷它們是否能生成同樣的二叉搜索樹。
輸入格式:
輸入包含若干組測試數據。每組數據的第1行給出兩個正整數N (≤10)和L,分別是每一個序列插入元素的個數和須要檢查的序列個數。第2行給出N個以空格分隔的正整數,做爲初始插入序列。最後L行,每行給出N個插入的元素,屬於L個須要檢查的序列。
簡單起見,咱們保證每一個插入序列都是1到N的一個排列。當讀到N爲0時,標誌輸入結束,這組數據不要處理。
輸出格式:
對每一組須要檢查的序列,若是其生成的二叉搜索樹跟對應的初始序列生成的同樣,輸出「Yes」,不然輸出「No」。
輸入樣例:
4 2 3 1 4 2 3 4 1 2 3 2 4 1 2 1 2 1 1 2 0
輸出樣例:
Yes No No
本題要求咱們對於輸入的各類插入序列,判斷它們是否能生成同樣的二叉搜索樹。在輸入樣例中包含三部份內容。第一部分是第一行的兩個整數:4表示插入序列所包含的個數,即二叉樹的結點個數,2表示後面有兩個序列須要取比較和前面是否同樣;第二部分是輸入的序列;第三部分就是後面輸入的若干組序列,它們要和第一組序列作比較。
這道題其實是兩個序列是否對應相同搜索樹的判別。
#include <iostream> #include <vector> using namespace std; typedef struct node Node; struct node{ int left; int right; }; //初始化二叉樹函數 void Init_Tree(vector<Node> &Tree,int N) { for ( int i = 1 ; i <= N ; i++){ Tree[i].left = -1; Tree[i].right = -1; } } //建樹函數 void Build_Tree(vector<Node> &Tree,int N) { int value; int flag = 0; int root = 0; int pre = 0; while(N--){ cin>>value; if ( flag == 0){ root = value; pre = root; flag = 1; }else{ while(1){ //當前輸入值比訪問的上一個結點pre(pre最初爲根結點)大,且pre有右孩子 if (value > pre && Tree[pre].right != -1){ pre = Tree[pre].right; } //當前輸入值比訪問的上一個結點pre(pre最初爲根結點)大,且pre無右孩子 if (value > pre && Tree[pre].right == -1){ Tree[pre].right = value; pre = root;//下一次輸入數字也從根結點開始比較 break; } //當前輸入值比訪問的上一個結點pre(pre最初爲根結點)小,且pre有左孩子 if (value<pre && Tree[pre].left != -1) { pre=Tree[pre].left; } //當前輸入值比訪問的上一個結點pre(pre最初爲根結點)小,且pre無左孩子 if (value<pre && Tree[pre].left == -1) { Tree[pre].left=value; pre=root;//下一次輸入數字也從根結點開始比較 break; } } } } } //比較兩棵二叉搜索樹是否相同的函數 bool Compare_Tree(vector<Node> &Tree1,vector<Node> &Tree2 ,int N) { bool flag = true; for ( int i = 1 ; i <= N ; i++){ if (!(Tree1[i].left == Tree2[i].left && Tree1[i].right == Tree2[i].right)){ flag = false; break; } } return flag; } int main() { int N,L; int flag = 0; while(1){ cin>>N; if ( N == 0){ break; } cin>>L; vector<vector<Node>> Tree(L,vector<Node>(11)); vector<Node> tree(11); Init_Tree(tree,N); for ( int i = 0 ; i < L ; i++){ Init_Tree(Tree[i],N); } Build_Tree(tree,N); for ( int i = 0 ; i < L ; i++){ Build_Tree(Tree[i],N); if (Compare_Tree(tree,Tree[i],N)){ if ( flag == 0){ flag = 1; cout<<"Yes"; }else{ cout<<"\n"<<"Yes"; } }else{ if ( flag == 0){ flag = 1; cout<<"No"; }else{ cout<<"\n"<<"No"; } } } } return 0; }
An AVL tree is a self-balancing binary search tree. In an AVL tree, the heights of the two child subtrees of any node differ by at most one; if at any time they differ by more than one, rebalancing is done to restore this property. Figures 1-4 illustrate the rotation rules.
Now given a sequence of insertions, you are supposed to tell the root of the resulting AVL tree.
Input Specification:
Each input file contains one test case. For each case, the first line contains a positive integer N (≤20) which is the total number of keys to be inserted. Then N distinct integer keys are given in the next line. All the numbers in a line are separated by a space.
Output Specification:
For each test case, print the root of the resulting AVL tree in one line.
Sample Input 1:
5 88 70 61 96 120
Sample Output 1:
70
Sample Input 2:
7 88 70 61 96 120 90 65
Sample Output 2:
88
#include<iostream> using namespace std; typedef int ElemType; typedef struct AVLTreeNode *AVLTree; struct AVLTreeNode { ElemType data; AVLTree left; AVLTree right; int height; }; int GetHeight(AVLTreeNode *tree) { if (tree == NULL) return -1; //空樹返回-1 else return tree->height; } int Max(int a,int b) { if (a > b) return a; else return b; } AVLTree SingleLeftRotation(AVLTree A) { /* 注意:A 必須有一個左子結點 B */ /* 將 A 與 B 作如圖 4.35 所示的左單旋,更新 A 與 B 的高度,返回新的根結點 B */ AVLTree B = A->left; A->left = B->right; B->right = A; A->height = Max(GetHeight(A->left), GetHeight(A->right)) + 1; B->height = Max(GetHeight(B->left), A->height) + 1; return B; } AVLTree SingleRightRotation(AVLTree A) { /* 注意:A 必須有一個左子結點 B */ /* 將 A 與 B 作如圖 4.35 所示的右單旋,更新 A 與 B 的高度,返回新的根結點 B */ AVLTree B = A->right; A->right = B->left; B->left = A; A->height = Max(GetHeight(A->right), GetHeight(A->left)) + 1; B->height = Max(GetHeight(B->right), A->height) + 1; return B; } AVLTree DoubleLeftRightRotation(AVLTree A) { /* 注意:A 必須有一個左子結點 B,且 B 必須有一個右子結點 C */ /* 將 A、B 與 C 作如圖 4.38 所示的兩次單旋,返回新的根結點 C */ A->left = SingleRightRotation(A->left); /*將 B 與 C 作右單旋,C 被返回*/ return SingleLeftRotation(A); /*將 A 與 C 作左單旋,C 被返回*/ } AVLTree DoubleRightLeftRotation(AVLTree A) { /* 注意:A 必須有一個左子結點 B,且 B 必須有一個右子結點 C */ /* 將 A、B 與 C 作如圖 4.38 所示的兩次單旋,返回新的根結點 C */ A->right = SingleLeftRotation(A->right); /*將 B 與 C 作右單旋,C 被返回*/ return SingleRightRotation(A); /*將 A 與 C 作左單旋,C 被返回*/ } AVLTree AVL_Insertion(ElemType X, AVLTree T) { /* 將 X 插入 AVL 樹 T 中,而且返回調整後的 AVL 樹 */ if (!T) { /* 若插入空樹,則新建包含一個結點的樹 */ T = (AVLTree)malloc(sizeof(struct AVLTreeNode)); T->data = X; T->height = 0; T->left = T->right = NULL; } /* if (插入空樹) 結束 */ else if (X < T->data) { /* 插入 T 的左子樹 */ T->left = AVL_Insertion(X, T->left); if (GetHeight(T->left) - GetHeight(T->right) == 2) /* 須要左旋 */ if (X < T->left->data) T = SingleLeftRotation(T); /* 左單旋 */ else T = DoubleLeftRightRotation(T); /* 左-右雙旋 */ } /* else if (插入左子樹) 結束 */ else if (X > T->data) { /* 插入 T 的右子樹 */ T->right = AVL_Insertion(X, T->right); if (GetHeight(T->left) - GetHeight(T->right) == -2) /* 須要右旋 */ if (X > T->right->data) T = SingleRightRotation(T); /* 右單旋 */ else T = DoubleRightLeftRotation(T); /* 右-左雙旋 */ } /* else if (插入右子樹) 結束 */ /* else X == T->Data,無須插入 */ T->height = Max(GetHeight(T->left), GetHeight(T->right)) + 1; /*更新樹高*/ return T; } int main() { int n; cin >> n; AVLTree root = NULL; int x; for (int i = 0; i < n; i++) { cin >> x; root = AVL_Insertion(x, root); } cout << root->data; return 0; }
將一系列給定數字插入一個初始爲空的小頂堆H[]。隨後對任意給定的下標i,打印從H[i]到根結點的路徑。
輸入格式:
每組測試第1行包含2個正整數N和M(≤1000),分別是插入元素的個數、以及須要打印的路徑條數。下一行給出區間[-10000, 10000]內的N個要被插入一個初始爲空的小頂堆的整數。最後一行給出M個下標。
輸出格式:
對輸入中給出的每一個下標i,在一行中輸出從H[i]到根結點的路徑上的數據。數字間以1個空格分隔,行末不得有多餘空格。
輸入樣例:
5 3 46 23 26 24 10 5 4 3
輸出樣例:
24 23 10 46 23 10 26 10
本題其實是一個最小堆查詢問題,在輸入樣例中給定5個數據構成一個最小堆,查詢3次。第二行的5個數據就構成了一個最小堆,第三行的5 4 3
分別表明最小堆中的下標。
#include <iostream> using namespace std; class MinHeap{ private : int* data; int capacity; int size; public: MinHeap(int N){ this->capacity = N; this->size = 0; this->data = new int[10000]; this->data[0] = -10000; } int GetSize(){ return this->size; } bool IsFull(){ return this->size == this->capacity; } bool IsEmpty(){ return this->size == 0; } void Insert(int data){ if ( this->IsFull()){ return ; } int i = ++this->size; for ( ; this->data[i/2] > data ; i /= 2){ this->data[i] = this->data[i/2]; } this->data[i] = data; } void Find_Path(int index){ if (index > this->size){ return; } bool flag = false; for ( int i = index ; i >= 1 ; i /= 2){ if (!flag){ cout<<this->data[i]; flag = true; }else{ cout<<" "<<this->data[i]; } } } }; int main() { int N,L; cin>>N>>L; MinHeap minheap(N); for ( int i = 1 ; i <= N ; i++){ int data; cin>>data; minheap.Insert(data); } for ( int i = 0 ; i < L ; i++){ int index; cin>>index; minheap.Find_Path(index); cout<<"\n"; } return 0; }
參考連接:
03-樹1 樹的同構 (25分)
PTA List Leaves
04-樹4 是否同一棵二叉搜索樹 (25分)
Root of AVL Tree
05-樹7 堆中的路徑 (25分)
不足之處,歡迎指正。