面试常考之二叉树

二叉树在数据结构面试中的地位举足轻重,算得上是大公司面试必问,笔试必考;因为对二叉树的操作直接反应一个人的数据结构功底有多深厚,基础知识是否扎实。。。(一点废话),下面就二叉树的基本操作说一说二叉树的知识点,不对之处还请指正。

面试常考的几个操作:

1:二叉树的基本性质

2:递归建立二叉树

3:递归遍历二叉树(先序,中序,后序)

4:非递归遍历二叉树(先序,中序,后序)

5:求二叉树中的节点个数

6:求二叉树的深度

7:分层遍历二叉树

8:求二叉树第K层的节点个数

9:求二叉树的镜像

10:测试源码

下面就常用的算法给大家讲解这些二叉树的基本性质。

说明:

所有代码使用C语言编写,头信息,常量如下:

 1 **
 2  * 树的基本操作tree
 3  */
 4 #include <stdio.h>
 5 #include <stdlib.h>
 6 #include <string.h>
 7 
 8 #define ERROR -1
 9 #define OK 1
10 #define OVERFLOW 2
11 #define STACK_INIT_SIZE 100
12 #define STACKINCREMENT 20
13 
14 //最大队列长度
15 #define MAXQSIZE 100
16 
17 typedef int Struct;
18 typedef char TElement;
19 typedef int Status;
20 int i = 0; //记录树初始化过程,数据的偏移
21 int j = 0; //记录树初始化过程,数据的偏移

二叉树结构体,以及栈、队列的结构体(层次遍历,非递归要用到)如下:

 1 //二叉树结构体(二叉链表存储结构)
 2 typedef struct BiNode{
 3     TElement data;
 4     struct BiNode *lchild,*rchild;
 5 }BiTNode, *BiTree;
 6 
 7 //栈表示(用于非递归)
 8 typedef struct{
 9     BiTree *base;
10     BiTree *top;
11     int stacksize;
12 }SqStack;
13 
14 //使用队列,实现层次遍历
15 typedef struct QNode{
16     BiTree data;
17     struct QNode *next;
18 }QNode, *QueuePtr;
19 
20 typedef struct{
21     QueuePtr front;
22     QueuePtr rear;
23 }LinkQueue;

 

1:二叉树的基本性质 

一、二叉树第 i 层上最多有2i-1个节点

二、深度为 k 的二叉树,最多有2k - 1个节点

三、对于任意一颗二叉树T,如果其终端节点数为n1度数为2的节点数为n2 则 n1 = n2 + 1;

四、具有n个节点的完全二叉树深度为[ log2n] + 1;

 

2:递归建立二叉树

递归常见二叉树,先从左子树开始,arr 是调用传值过去的节点值,i 用于移动数组内的节点值

//创建二叉树
Struct CreateBiTree(BiTree &T, char arr[]){
    if(arr[i] == ' '){
        i++;
        T = NULL;
    }else{
        T = (BiTree)malloc(sizeof(BiNode));
        if(!T){
            exit(ERROR);//分配空间失败;
        }
        T->data = arr[i];
        i++;
        CreateBiTree(T->lchild, arr);
        CreateBiTree(T->rchild, arr);
    }
    return OK;
}

 

3:递归遍历二叉树(先序,中序,后序)

这是三中常规的遍历二叉树的方法,用的非常多,不仅便于理解,而且三个函数只是调用顺序不同,所以书上页数先给出了这三种遍历。当然在面试中也长问到。

 1 //先序遍历(递归方法)
 2 int PrintTree(BiTree T){
 3     if(T){
 4         printf("%c ",T->data);
 5         PrintTree(T->lchild);
 6         PrintTree(T->rchild);
 7         
 8     }
 9     return OK;
10 }
11 
12 //中序遍历(递归方法)
13 int PrintTreeL(BiTree T){
14     if(T){
15         PrintTreeL(T->lchild);
16         printf("%c ",T->data);
17         PrintTreeL(T->rchild);
18     }
19     return OK;
20 }
21 
22 //后序遍历(递归方法)
23 int PrintTreeR(BiTree T){
24     if(T){
25         PrintTreeR(T->lchild);
26         PrintTreeR(T->rchild);
27         printf("%c ",T->data);
28     }
29     return OK;
30 }

 

4:非递归遍历二叉树(先序,中序,后序)

非递归遍历在面试中问的非常多,特别是大公司的面试,几乎是必问,所以这里就非递归的方法每个给出了两种

一、先序遍历:

  先序遍历这里给出了两个方法,两种方法都是借助于栈来实现,具体的栈的操作在最后源程序中都有,看家可以看看,非递归的操作比递归麻烦得多,第一种方法是:首先根节点入栈,然后while循环一致判断栈是否为空,第二个while循环一直讲左节点入栈,知道左子树为空。其中PopN(S);  方法是为了将多余的空元素弹出。第二种方法比较推荐开始根节点不入栈,先判断根节点是否为空,这样栈空间中不会有空元素,效率高一点,并且便于理解

 1 //先序遍历(非递归方法)(方法1)
 2 int PrintTree_Re(BiTree T){
 3     SqStack S;
 4     InitStack(S);//建立栈
 5     Push(S,T); //树的根节点先入栈
 6     BiTree P;
 7     while(StackEmpty(S) == 1){
 8         while(GetTop(S,P) && P){
 9             printf("%c ",P->data);
10             Push(S,P->lchild);    
11         }
12         PopN(S); 
13         if(StackEmpty(S) == 1){
14             Pop(S,P);
15             Push(S,P->rchild);
16         }
17     }
18     return OK;
19 }
20 
21 //先序遍历(非递归方法)(方法2)
22 int PrintTree_Re_T(BiTree T){
23     SqStack s;
24     InitStack(s);
25     BiTree p;
26     p = T;
27     while(p || StackEmpty(s) == 1){
28         while(p){
29             printf("%c ",p->data);
30             Push(s,p);    
31             p = p->lchild;
32         }
33         if(StackEmpty(s) == 1){
34             Pop(s,p);
35             p = p->rchild;
36         }
37     }
38     return OK;
39 }

二、中序遍历:

中序遍历这里也给出2种方法。感觉和先序差不多,只是遍历顺序不同,处理的时候方式一样,这里不再多说。

 1 //中序遍历(非递归)(方法1)
 2 int PrintTreeL_Re(BiTree T){
 3     SqStack S;
 4     InitStack(S);//建立栈
 5     Push(S,T); //树的根节点先入栈
 6     BiTree P;
 7     while(StackEmpty(S) == 1){
 8         while(GetTop(S,P) && P){
 9             Push(S,P->lchild);    
10         }
11         PopN(S); 
12         if(StackEmpty(S) == 1){
13             Pop(S,P);
14             printf("%c ",P->data);
15             Push(S,P->rchild);
16         }
17     }
18     return OK;
19 }
20 
21 //中序遍历(非递归)(方法2)
22 int PrintTreeL_Re_T(BiTree T){
23     SqStack S;
24     InitStack(S);//建立栈
25     BiTree P;
26     P = T;
27     while(P || StackEmpty(S) == 1){
28         if(P){
29             Push(S,P);
30             P = P->lchild;
31         }else{
32             Pop(S,P);
33             printf("%c ",P->data);
34             P = P->rchild;
35         }
36     }
37     return OK;
38 }

二、后序遍历:

个人感觉后序遍历的非递归是三种遍历方式中最难的。写了很长时间,总感觉找不到一种合适的方法来保证孩子节点完全遍历。下面来说说这种非递归的遍历。其实会了也比较简单,对于结点P,要保证根结点在左孩子和右孩子访问之后才能访问,先将其入栈。如果P不存在左孩子和右孩子,可以直接访问;或者P 存在左孩子或者右孩子,访问过左孩子和右孩子后可以直接访问该结点。其他情况,则将P的右孩子和左孩子依次入栈。这个就会这一种,应该有其他方法,欢迎指出

 1 //后序遍历(非递归方法)
 2 int PrintTreeR_Re(BiTree T){
 3     SqStack s;
 4     InitStack(s);//建立栈
 5     Push(s,T); //树的根节点先入栈
 6     BiTree p,q;
 7     while(StackEmpty(s) == 1){
 8         GetTop(s,p);
 9         if((p->lchild == NULL && p->rchild == NULL) || ( q && ( q == p->lchild || q == p->rchild))){
10             printf("%c ",p->data);
11             PopN(s); 
12             q = p;
13         }else{
14             if(p->rchild){
15                 Push(s,p->rchild);
16             }
17             if(p->lchild){
18                 Push(s,p->lchild);
19             }
20         }
21     }
22     return OK;
23 }

 

5:求二叉树中的节点个数

其实就是遍历一遍二叉树,不再多说,看程序便知

 1 //统计二叉树节点的个数
 2 void getTreeNodeNum(BiTree T,int &num){
 3     if(!T){
 4         num = 0;
 5     }
 6     num++;
 7     if(T->lchild){
 8         getTreeNodeNum(T->lchild,num);
 9     }
10     if(T->rchild){
11         getTreeNodeNum(T->rchild,num);
12     }
13 }

 

6:求二叉树的深度

思想就是从左子树开始遍历,递归完整个树结构,如果二叉树不为空,二叉树的深度 = max(左子树深度, 右子树深度) + 1,参考代码如下:

 1 //得到树的深度
 2 int getTreeHeight(BiTree T){
 3     int hl,hr,max;
 4     if(T!=NULL){
 5         hl=getTreeHeight(T->lchild);
 6         hr=getTreeHeight(T->rchild);
 7         max=hl>hr?hl:hr;
 8         return (max+1);
 9     }else{
10         return 0;
11     }
12 }

 

7:分层遍历二叉树

一、树的层次结构分层打印与队列的的先进先出相似,所以我选择用队列来完成,首先根节点入队列,然后左孩子,又孩子以此入队列,打印即可

 1 /*层次遍历二叉树(借助队列实现)*/
 2 void HierarchicalTree_Q(BiTree T){
 3     LinkQueue Q;
 4     InitQueue(Q); //建立队列
 5     EnQueue(Q,T); //树的根节点先入列
 6     BiTree Ta;
 7     while(Q.front != Q.rear){
 8         DeQueue(Q,Ta);
 9         if(Ta->lchild){
10             EnQueue(Q,Ta->lchild);
11         }
12         if(Ta->rchild){
13             EnQueue(Q,Ta->rchild);
14         }
15         printf("%c ",Ta->data);
16     }
17 }

二、说到分层如果打印树的真实结构应该也可以,所以我想借助堆栈来实现,有兴趣的可以看一下,思想:其实和层次遍历一样,最难的是怎样控制换行先,我的想法是创建一个标志位*,当前行入栈完毕就插入标志位*遍历一个节点就把左孩子和右孩子入栈,以此一层一层的打印,代码检验过,不知道还有没有错误,欢迎指正

 1 /*层次遍历二叉树(并且分层打印)(借助栈实现)*/
 2 void  HierarchicalTree_DA(BiTree T){
 3     LinkQueue Q;
 4     InitQueue(Q);//建立队列
 5     EnQueue(Q,T);//树的根节点先入列
 6     BiTree Ta1,Tb1,Tc1;
 7     
 8     //创建标志位
 9     char data[3] = {'*',' ',' '};
10     CreateBiTree(Ta1,data);
11     
12     //创建移动的指针
13     QueuePtr point;
14     point = Q.front->next;//移动指针先指向头结点
15     EnQueue(Q,Ta1);//插入标志位
16     printf("\n");
17     while(point){
18         //判断左右节点
19         if(point->data->lchild || point->data->rchild){
20             if(point->data->lchild){
21                 EnQueue(Q,point->data->lchild);
22             }
23             if(point->data->rchild){
24                 EnQueue(Q,point->data->rchild);
25             }
26         }    
27         point = point->next; //指针下移一位
28         if(point->data->data == '*'){
29             if(!point->next){//判断是否到结尾(到结尾直接退出)
30                 break;
31             }
32             EnQueue(Q,Ta1);//插入标志节点
33             point = point->next; //如果遇到标志位,则指针下移
34         }
35     }
36     
37     printf("\n");
38     //循环输出节点
39     while(Q.front != Q.rear){
40         DeQueue(Q,Tc1);
41         char str = Tc1->data;
42         if(str == '*'){
43             printf("\n");
44         }else{
45             printf("%c ",str);
46         }
47     }
48 }

 

8:求二叉树第K层的节点个数

利用递归的次数来(孩子节点不为空的条件)来简介得到k层节点的个数,比较简单,代码如下

 1 //得到第k层的节点个数
 2 int getNodeNum(BiTree T, int k){
 3     if (!T || k < 1){
 4         return 0;
 5     }
 6     if (k == 1){
 7         return 1;
 8     }
 9     return getNodeNum(T->lchild, k-1) + getNodeNum(T->rchild, k-1);
10 }

 

9:求二叉树的镜像

所谓的镜像其实就是左子树和右子树互换颠倒,如下面代码。

 1 //写出二叉树的镜像
 2 void getNoClone(BiTree &T){
 3     BiTree Ta;
 4     if(T){
 5         Ta = T->lchild;
 6         T->lchild = T->rchild;
 7         T->rchild = Ta;
 8         getNoClone(T->lchild);
 9         getNoClone(T->rchild);
10     }
11 }

 

10:测试源码

最后附上完整的测试代码,其中包含了,队列的初始化,插入队列,出队列操作,栈的初始化,进栈,出栈,得到栈顶元,树的初始化素等也可以复习一下。中间可定有不对不足的地方,还望大神们多多指正

  1 /**
  2  * 树的基本操作tree
  3  */
  4 #include <stdio.h>
  5 #include <stdlib.h>
  6 #include <string.h>
  7 
  8 #define ERROR -1
  9 #define OK 1
 10 #define OVERFLOW 2
 11 #define STACK_INIT_SIZE 100
 12 #define STACKINCREMENT 20
 13 
 14 //最大队列长度
 15 #define MAXQSIZE 100
 16 
 17 typedef int Struct;
 18 typedef char TElement;
 19 typedef int Status;
 20 int i = 0; //记录树初始化过程,数据的偏移
 21 int j = 0; //记录树初始化过程,数据的偏移
 22 
 23 
 24 //二叉树结构体(二叉链表存储结构)
 25 typedef struct BiNode{
 26     TElement data;
 27     struct BiNode *lchild,*rchild;
 28 }BiTNode, *BiTree;
 29 
 30 //栈表示
 31 typedef struct{
 32     BiTree *base;
 33     BiTree *top;
 34     int stacksize;
 35 }SqStack;
 36 
 37 //使用队列,实现层次遍历
 38 typedef struct QNode{
 39     BiTree data;
 40     struct QNode *next;
 41 }QNode, *QueuePtr;
 42 
 43 typedef struct{
 44     QueuePtr front;
 45     QueuePtr rear;
 46 }LinkQueue;
 47 
 48 
 49 //初始化队列
 50 Status InitQueue(LinkQueue &Q){
 51     Q.front = Q.rear = (QueuePtr)malloc(sizeof(BiTree));//动态分配空间
 52     if(!Q.front){
 53         exit(ERROR);//分配空间失败
 54     }
 55     Q.front->next = NULL;
 56     return OK;
 57 }
 58 
 59 //在队尾插入新元素
 60 Status EnQueue(LinkQueue &Q, BiTree e){
 61     QueuePtr p;
 62     p = (QueuePtr)malloc(sizeof(BiTree));
 63     if(!p){
 64         exit(ERROR);//分配失败
 65     }
 66     p->data = e;
 67     p->next = NULL;
 68     Q.rear->next = p;
 69     Q.rear = p;
 70     return OK;
 71 }
 72 
 73 //删除队头元素,并用e返回
 74 void DeQueue(LinkQueue &Q,BiTree &e){
 75     
 76     if(Q.front != Q.rear){//先判断队列是否为空
 77         QueuePtr p;
 78         e = Q.front->next->data;
 79         if(Q.front->next == Q.rear){//队列只有一个元素
 80             p = Q.rear;
 81             Q.rear = Q.front;
 82             Q.front->next = NULL;
 83         }else{
 84             p = Q.front->next;
 85             Q.front->next = p->next;
 86             p->next = NULL;
 87         }
 88         //free(p->data);
 89     }
 90 }
 91 
 92 //创建二叉树
 93 Struct CreateBiTree(BiTree &T, char arr[]){
 94     if(arr[i] == ' '){
 95         i++;
 96         T = NULL;
 97     }else{
 98         T = (BiTree)malloc(sizeof(BiNode));
 99         if(!T){
100             exit(ERROR);//分配空间失败;
101         }
102         T->data = arr[i];
103         i++;
104         CreateBiTree(T->lchild, arr);
105         CreateBiTree(T->rchild, arr);
106     }
107     return OK;
108 }
109 
110 //先序遍历(递归方法)
111 int PrintTree(BiTree T){
112     if(T){
113         printf("%c ",T->data);
114         PrintTree(T->lchild);
115         PrintTree(T->rchild);
116         
117     }
118     return OK;
119 }
120 
121 //中序遍历(递归方法)
122 int PrintTreeL(BiTree T){
123     if(T){
124         PrintTreeL(T->lchild);
125         printf("%c ",T->data);
126         PrintTreeL(T->rchild);
127     }
128     return OK;
129 }
130 
131 //后序遍历(递归方法)
132 int PrintTreeR(BiTree T){
133     if(T){
134         PrintTreeR(T->lchild);
135         PrintTreeR(T->rchild);
136         printf("%c ",T->data);
137     }
138     return OK;
139 }
140 
141 
142 //初始化栈
143 Struct InitStack(SqStack &S){
144     S.base = (BiTree *)malloc(STACK_INIT_SIZE * sizeof(BiTree));
145     S.top = S.base;
146     if(!S.base){
147         exit(ERROR);//分配失败
148     }
149     S.stacksize = STACK_INIT_SIZE;
150     return OK;
151 }
152 
153 //push入栈操作
154 Struct Push(SqStack &S, BiTree T){
155     if(S.top - S.base >= S.stacksize){
156         S.base = (BiTree *)realloc(S.base,(S.stacksize + STACKINCREMENT) * sizeof(BiTree));
157         if(!S.base){
158             exit(ERROR);
159         }
160         S.top = S.base + S.stacksize;
161         S.stacksize += STACKINCREMENT;
162     }    
163     *S.top = T;
164     S.top++;
165     
166     return OK;
167 }
168 
169 //出栈操作
170 Struct Pop(SqStack &S,BiTree &P){
171     if(S.base != S.top){
172         P = *(--S.top);
173     }
174     return OK;
175 }
176 
177 //出栈操作
178 Struct PopN(SqStack &S){
179     if(S.base != S.top){
180         --S.top;
181     }
182     return OK;
183 }
184 
185 //得到栈顶元素操作
186 Struct GetTop(SqStack S,BiTree &P){
187     if(S.base != S.top){
188         P = *(S.top - 1);
189     }
190     return OK;
191 }
192 
193 //判断是否是空栈
194 int StackEmpty(SqStack S){
195     if(S.base != S.top){
196         return 1;
197     }else{
198         return 0;
199     }
200 }
201 
202 //返回栈长度
203 int GetLength(SqStack S){
204     return S.top - S.base;
205 }
206 
207 //先序遍历(非递归方法)(方法1)
208 int PrintTree_Re(BiTree T){
209     SqStack S;
210     InitStack(S);//建立栈
211     Push(S,T); //树的根节点先入栈
212     BiTree P;
213     while(StackEmpty(S) == 1){
214         while(GetTop(S,P) && P){
215             printf("%c ",P->data);
216             Push(S,P->lchild);    
217         }
218         PopN(S); 
219         if(StackEmpty(S) == 1){
220             Pop(S,P);
221             Push(S,P->rchild);
222         }
223     }
224     return OK;
225 }
226 
227 //先序遍历(非递归方法)(方法2)
228 int PrintTree_Re_T(BiTree T){
229     SqStack s;
230     InitStack(s);
231     BiTree p;
232     p = T;
233     while(p || StackEmpty(s) == 1){
234         while(p){
235             printf("%c ",p->data);
236             Push(s,p);    
237             p = p->lchild;
238         }
239         if(StackEmpty(s) == 1){
240             Pop(s,p);
241             p = p->rchild;
242         }
243     }
244     return OK;
245 }
246 
247 //中序遍历(非递归)(方法1)
248 int PrintTreeL_Re(BiTree T){
249     SqStack S;
250     InitStack(S);//建立栈
251     Push(S,T); //树的根节点先入栈
252     BiTree P;
253     while(StackEmpty(S) == 1){
254         while(GetTop(S,P) && P){
255             Push(S,P->lchild);    
256         }
257         PopN(S); 
258         if(StackEmpty(S) == 1){
259             Pop(S,P);
260             printf("%c ",P->data);
261             Push(S,P->rchild);
262         }
263     }
264     return OK;
265 }
266 
267 //中序遍历(非递归)(方法2)
268 int PrintTreeL_Re_T(BiTree T){
269     SqStack S;
270     InitStack(S);//建立栈
271     BiTree P;
272     P = T;
273     while(P || StackEmpty(S) == 1){
274         if(P){
275             Push(S,P);
276             P = P->lchild;
277         }else{
278             Pop(S,P);
279             printf("%c ",P->data);
280             P = P->rchild;
281         }
282     }
283     return OK;
284 }
285 
286 //后序遍历(非递归方法)
287 int PrintTreeR_Re(BiTree T){
288     SqStack s;
289     InitStack(s);//建立栈
290     Push(s,T); //树的根节点先入栈
291     BiTree p,q;
292     while(StackEmpty(s) == 1){
293         GetTop(s,p);
294         if((p->lchild == NULL && p->rchild == NULL) || ( q && ( q == p->lchild || q == p->rchild))){
295             printf("%c ",p->data);
296             PopN(s); 
297             q = p;
298         }else{
299             if(p->rchild){
300                 Push(s,p->rchild);
301             }
302             if(p->lchild){
303                 Push(s,p->lchild);
304             }
305         }
306     }
307     return OK;
308 }
309 
310 
311 /*层次遍历二叉树*/
312 void HierarchicalTree(BiTree T){
313     BiTree Ta = T;
314     if(Ta){
315         printf("%c ",Ta->data);
316         if(Ta->lchild){
317             //printf("%c ",Ta->lchild->data);
318             HierarchicalTree(Ta->lchild);
319         }
320         if(Ta->rchild){
321             //printf("%c ",Ta->rchild->data);
322             HierarchicalTree(Ta->rchild);
323         }
324     }
325 }
326 
327 /*层次遍历二叉树(借助队列实现)*/
328 void HierarchicalTree_Q(BiTree T){
329     LinkQueue Q;
330     InitQueue(Q); //建立队列
331     EnQueue(Q,T); //树的根节点先入列
332     BiTree Ta;
333     while(Q.front != Q.rear){
334         DeQueue(Q,Ta);
335         if(Ta->lchild){
336             EnQueue(Q,Ta->lchild);
337         }
338         if(Ta->rchild){
339             EnQueue(Q,Ta->rchild);
340         }
341         printf("%c ",Ta->data);
342     }
343 }
344 
345 /*层次遍历二叉树(并且分层打印)(借助队列实现)*/
346 void  HierarchicalTree_DA(BiTree T){
347     LinkQueue Q;
348     InitQueue(Q);//建立队列
349     EnQueue(Q,T);//树的根节点先入列
350     BiTree Ta1,Tb1,Tc1;
351     
352     //创建标志位
353     char data[3] = {'*',' ',' '};
354     CreateBiTree(Ta1,data);
355     
356     //创建移动的指针
357     QueuePtr point;
358     point = Q.front->next;//移动指针先指向头结点
359     EnQueue(Q,Ta1);//插入标志位
360     printf("\n");
361     while(point){
362         //判断左右节点
363         if(point->data->lchild || point->data->rchild){
364             if(point->data->lchild){
365                 EnQueue(Q,point->data->lchild);
366             }
367             if(point->data->rchild){
368                 EnQueue(Q,point->data->rchild);
369             }
370         }    
371         point = point->next; //指针下移一位
372         if(point->data->data == '*'){
373             if(!point->next){//判断是否到结尾(到结尾直接退出)
374                 break;
375             }
376             EnQueue(Q,Ta1);//插入标志节点
377             point = point->next; //如果遇到标志位,则指针下移
378         }
379     }
380     
381     printf("\n");
382     //循环输出节点
383     while(Q.front != Q.rear){
384         DeQueue(Q,Tc1);
385         char str = Tc1->data;
386         if(str == '*'){
387             printf("\n");
388         }else{
389             printf("%c ",str);
390         }
391     }
392 }
393 
394 //统计二叉树节点的个数
395 void getTreeNodeNum(BiTree T,int &num){
396     if(!T){
397         num = 0;
398     }
399     num++;
400     if(T->lchild){
401         getTreeNodeNum(T->lchild,num);
402     }
403     if(T->rchild){
404         getTreeNodeNum(T->rchild,num);
405     }
406 }
407 
408 //得到树的深度
409 int getTreeHeight(BiTree T){
410     int hl,hr,max;
411     if(T!=NULL){
412         hl=getTreeHeight(T->lchild);
413         hr=getTreeHeight(T->rchild);
414         max=hl>hr?hl:hr;
415         return (max+1);
416     }else{
417         return 0;
418     }
419 }
420 
421 //得到第k层的节点个数
422 int getNodeNum(BiTree T, int k){
423     if (!T || k < 1){
424         return 0;
425     }
426     if (k == 1){
427         return 1;
428     }
429     return getNodeNum(T->lchild, k-1) + getNodeNum(T->rchild, k-1);
430 }
431 
432 void main(){
433     i = 0;
434     //char Data[15] = {'A','B','C',' ',' ','D','E',' ','G',' ',' ','F',' ',' ',' '};
435     //char Data[15] = {'A','B','C',' ',' ','D','E',' ',' ','G',' ',' ','F',' ',' '};    
436     char Data[16] = {'A','B','C',' ',' ',' ','D','E',' ','G',' ',' ','F',' ',' ',' '};
437     //char Data[7] = {'A','B','C',' ',' ',' ',' '};
438     //char Data[7] = {'A',' ','b',' ','c',' ',' '};
439     BiTree Ta,Tb;
440     CreateBiTree(Ta,Data);
441     printf("*****************递归方法遍历二叉树**************\n");
442     printf("先序遍历:");
443     PrintTree(Ta);
444 
445     printf("\n");
446     printf("中序遍历:");
447     PrintTreeL(Ta);
448 
449     printf("\n");
450     printf("后序遍历:");
451     PrintTreeR(Ta);
452     printf("\n");
453     
454     printf("非递归先序遍历法1:");
455     PrintTree_Re(Ta);
456     printf("\n");
457     
458     printf("非递归先序遍历法2:");
459     PrintTree_Re_T(Ta);
460     printf("\n");
461 
462     printf("非递归中序遍历法1:");
463     PrintTreeL_Re(Ta);
464     printf("\n");
465 
466     printf("非递归中序遍历法2:");
467     PrintTreeL_Re_T(Ta);
468     printf("\n");
469     
470     printf("非递归后序遍历:");
471     PrintTreeR_Re(Ta);
472     printf("\n");
473     
474     printf("层次遍历:");
475     HierarchicalTree_Q(Ta);
476     printf("\n");
477     
478     printf("\n");
479     printf("层次遍历(分层显示):");
480     i = 0;
481     HierarchicalTree_DA(Ta);
482     printf("\n");
483     
484     printf("总结点个数:");
485     int TreeNum = 0;
486     getTreeNodeNum(Ta,TreeNum);
487     printf("%d ",TreeNum);
488     printf("\n");
489 
490     printf("树的深度:");
491     int TreeHeight = 0;
492     TreeHeight = getTreeHeight(Ta);
493     printf("%d ",TreeHeight);
494     printf("\n");
495 
496     printf("得到第3层的节点个数:");
497     printf("%d",getNodeNum(Ta,3));
498     printf("\n");
499 
500 }

 

运行结果:

面试常考之二叉树_第1张图片

 

转载请注明出处,谢谢!

 

你可能感兴趣的:(面试常考之二叉树)