HDOJ-1100 Trees made to order
一、题目链接
http://acm.hdu.edu.cn/showproblem.php?pid=1100
二、题目分析
对二叉树的所有形态顺序编号,编号规则是:节点数越多的编号越大;节点数相等,左子树节点数越多的越大;节点数相等,左子树节点数也相等,则依此规则比较右子树。
现给定一个正整数,依题目要求输出对应编号的二叉树形态。
三、求解思路
由题目输出格式要求,很直观地联想到使用深度优先搜索dfs,以当前二叉树对应的编号为条件,依次递归输出左子树和右子树。
1 int main(void) {
2 int n;
3 unsigned int arr[20] = {
0};
4
5 init(arr, 20);
6
7 while (scanf("%d", &n) != EOF && n != 0) {
8 dfs(n, arr, 20);
9 printf("\n");
10 }
11
12 return 0;
13 }
其中arr存储的是不同节点数对应的二叉树形态数,即arr[i]表示有i个节点的二叉树总共有多少种形态。
1 void dfs(int n, unsigned int *arr, int num) {
2
3 int left, right;
4
5 if (n == 1) {
6 printf("X");
7 return ;
8 }
9
10 getChildren(&left, &right, n, arr, num);
11
12 if (left != 0) {
13 printf("(");
14 dfs(left, arr, num);
15 printf(")");
16 }
17 printf("X");
18 if (right != 0) {
19 printf("(");
20 dfs(right, arr, num);
21 printf(")");
22 }
23 }
通过getChildren得到左右子树对应的编号,然后递归搜索即可。
计算左右子树对应序号的思路:
-
计算总节点数nall、左子树节点数nleft和右子树节点数nright;
-
在计算总节点数和左子树节点数的同时ÿ