这题用到了卡特兰数,比较麻烦。关于卡特兰数的基本概念百度一下你就知道。

使用卡特兰数对数组元素进行分组之后,需要具体计算一下要求的是第几组的第几个数,然后向下递归。

首先来看利用卡特兰数分组:

  从1开始前4个卡特兰数是 C[1]=1, C[2]=2, C[3]=5, C[4]=14  (C[0]也是有定义的,C[0]=1)

  于是我们把第1个元素归为第1组,第2,3个元素归为第2组,4,5,6,7,8归为第3组,9到22归为第4组,这样分组就完成了

  这里的每个元素,就对应着题目里的每一棵树。而卡特兰数的下标,即树的节点数

  所以1个节点的树有1棵,2个节点的树2棵,3个节点的树5棵,4个节点的树14棵

  比如现在我们求第17棵树在第几组,17-C[1]-C[2]-C[3]=9,因为9<=C[4],所以第17棵树肯定在4个节点的树的第9个位置

然后是对卡特兰数进行递归拆分:

  由卡特兰数的性质,C[i]=C[0]*C[i-1] + C[1]*C[i-2] + ... + C[i-2]*C[1] + C[i-1]C[0]

  我们找到所在组的位置后,就要寻找向下递归的参数。

  比如我们刚才找到了第17棵树在C[4]的第9个位置,接下来要找这第9个位置的树究竟是什么

  首先,C[4]=C[0]*C[3] + C[1]*C[2]+C[2]*C[1] +C[3]*C[0],这相当于把C[4]所含的14棵树再一次进行了分组

  它使用规模更小的卡特兰数的乘积,对原先的卡特兰数进行分组,把C[4]分成C[0]*C[3]=5,C[1]*C[2]=2,C[2]*C[1]=2,C[3]*C[0]=5,一共4组

  那么第9棵树在哪呢?9-C[0]*C[3]-C[1]*C[2]=2,因为2<=C[2]*C[1],所以这第9棵树肯定在C[2]*C[1]那个分组里的第2棵

  C[2]就是左子树,由2个树节点构成。C[1]是右子树,由1个树节点构成。

  接下来应该对左右子树进行递归。递归结束条件是C[1],根据题意,找到C[1],就可以输出一个X

  但现在我们还缺少一些递归参数,右子树C[1]没问题,反正就输出X。但右子树是由2个节点构成的树,C[2]=2,所以这样的树有2棵,究竟是哪一棵?

  这里需要停下来想一想,对于这种卡特兰数形式的树来说,左子树x个节点,右子树y个节点,其内涵是什么?

    -  首先这是一棵由x+y+1个节点构成的树,对吧?

    -  其次这棵树一共有几种变化?C[x]*C[y]种

      -  比如x=2,y=3,那么这棵树一共有C[2]*C[3]=2*5=10种变化,具体来看:

        - 3个节点的右子树有5种变化,每当右子树5种变化结束之后,左子树才变化一次,我们把这样叫一轮变化。一共有C[2]=2轮,每轮5种,共2*5=10种变化  

         - 假设我们要找这10种变化里的第6种变化,那么是第几轮的第几种变化?应该是第二轮的第一种,对吧?

            - 如何用公式来表示呢?

          - 我们相当于用右子树的C[3]=5把10种变化分成了2轮,那么求第n种变化是第几轮第几种,公式为:第(n-1)/C[3]+1=轮的第(n-1)%C[3]+1种变化

          - 当n=6,(n-1)/C[3]+1=2,(n-1)%C[3]+1=1,即第2轮第1种

          - 即左子树C[2]里面的第2棵树,右子树C[3]里的第1棵树

  好了,这样我们已经找齐所有的递归参数,可以对左右子树分别进行递归。

  贴一下AC代码,JAVA版的:

import java.util.Scanner;

public class POJ1095 {

    static long[] catalan = new long[30];

    static void initCatalan(){
catalan[0]=catalan[1]=1;
for(int i=2;i<catalan.length;i++){
catalan[i] = (4*i-2)*catalan[i-1]/(i+1);
}
} // n个节点的第k种情况
static void draw(int n,long k){
if(n==0)
return;
if(n ==1){
System.out.print("X");
return;
}
// cn:去掉根节点后的节点数
int cn=n-1,leftn=0,rightn=0;
long rightc=0,leftk=0,rightk=0;
for(int i=cn;i>=0;i--){
long j = catalan[i]*catalan[cn-i];
if(j<k){
k-=j;
}else {
leftn = cn-i;
rightn = i;
rightc = catalan[i];
break;
}
}
leftk=(k-1)/rightc +1;
rightk=(k-1)%rightc +1;
if(leftn>0) {
System.out.print("(");
draw(leftn, leftk);
System.out.print(")");
}
System.out.print("X");
if(rightn>0) {
System.out.print("(");
draw(rightn, rightk);
System.out.print(")");
}
} public static void main(String[] args) {
initCatalan();
Scanner sc = new Scanner(System.in);
long c = Integer.valueOf(sc.nextLine());
int i;
while (c!=0) {
for(i=1;i<catalan.length;i++){
if(catalan[i]<c){
c-=catalan[i];
}else
break;
}
draw(i,c);
System.out.println();
c = Integer.valueOf(sc.nextLine());
}
}
}

POJ1095 Trees Made to Order(JAVA)的更多相关文章

  1. POJ 1095 Trees Made to Order 最详细的解题报告

    题目来源:Trees Made to Order 题目大意:根据下面的规则给一棵二叉树编号: 规则1:如果二叉树为空,则编号为0: 规则2:如果二叉树只有一个节点,则编号为1: 规则3:所有含有m个节 ...

  2. Trees Made to Order——Catalan数和递归

    Trees Made to Order Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 7155   Accepted: 40 ...

  3. HDU.P1100 Trees Made to Order 解题报告

    http://www.cnblogs.com/keam37/p/3637717.html  keam所有 转载请注明出处 Problem Description We can number binar ...

  4. POJ 1095 Trees Made to Order(卡特兰数列)

    题目链接 中间计算的各种细节.有的细节没处理好,就wa了...主要思路就是根据卡特兰数列的: h(n)= h(0)*h(n-1)+h(1)*h(n-2) + ... + h(n-1)h(0) (n&g ...

  5. poj 1095 Trees Made to Order 卡特兰数

    这题用到了卡特兰数,详情见:http://www.cnblogs.com/jackge/archive/2013/05/19/3086519.html 解体思路详见:http://blog.csdn. ...

  6. poj 1095 Trees Made to Order

    http://poj.org/problem?id=1095 先求出n个节点数的二叉树的形态有多少种.卡特兰数f[n]=f[n-1]*(4*n-2)/(n+1);再递归求. #include < ...

  7. HDOJ-1100 Trees made to order

    一.题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1100 二.题目分析 对二叉树的所有形态顺序编号,编号规则是:节点数越多的编号越大:节点数相等,左子 ...

  8. POJ题目排序的Java程序

    POJ 排序的思想就是根据选取范围的题目的totalSubmittedNumber和totalAcceptedNumber计算一个avgAcceptRate. 每一道题都有一个value,value ...

  9. ssh整合问题总结--运行项目时报java.lang.StackOverflowError(堆栈溢出)异常

    今天在整合ssh项目中,碰到一个异常,当我提交购物车数据到订单时,浏览器报了一个这样的异常. 当时,我就吓坏了.尼玛,这不是内存溢出了吗?吓得我赶紧去检查了每一个有遍历语句的代码,结果没有发现一个死循 ...

随机推荐

  1. PhpStorm 合理标注目录让索引和扫描更加地高效

    在项目开发过程中,以下几种情况让 PhpStorm 加重了索引扫描的负担: 1.npm install 之后产生的 node_modules 目录 2.runtime 运行时产生的目录 3.stora ...

  2. Laravel/Homestead storage:link -> symlink(): Protocol error

    I'm trying to run the following artisan command: php artisan storage:link I get this error: [ErrorEx ...

  3. Java 设计模式系列(十五)迭代器模式(Iterator)

    Java 设计模式系列(十五)迭代器模式(Iterator) 迭代器模式又叫游标(Cursor)模式,是对象的行为模式.迭代子模式可以顺序地访问一个聚集中的元素而不必暴露聚集的内部表象(interna ...

  4. centos7下查看tomcat是否启动/系统日志等

    centos7下查看tomcat是否启动/系统日志等  方法一: 首先,进入Tomcat下的bin目录 cd /usr/local/tomcat/bin 使用Tomcat关闭命令 ./shutdown ...

  5. Devexpress + wcf +ef 批量更新处理

    项目结构: 1.客户端:Winform, 2.数据访问:EF4.0(从数据库生成模型-懒人必需这样) 3.DTO:直接使用EF实体 4.服务端:WCF 托管在IIS中采用basicHttp帮定(这样可 ...

  6. python3 二叉树的存储和三种遍历

    #coding:utf-8 class node(): def __init__(self,k=None,l=None,r=None): self.key=k; self.left=l; self.r ...

  7. Kali linux切换语言为中文

    echo LANG="zh_CN.UTF-8" > /etc/default/locale

  8. C/C++编程可用的Linux自带工具

    GNU Binary Utilities或binutils是一整套的编程语言工具程序,用来处理许多格式的目标文件.当前的版本原本由在Cygnus Solutions的程序员以Binary File D ...

  9. flume学习以及ganglia(若是要监控hive日志,hive存放在/tmp/hadoop/hive.log里,只要运行过hive就会有)

    python3.6hdfs的使用 https://blog.csdn.net/qq_29863961/article/details/80291654 https://pypi.org/  官网直接搜 ...

  10. [label][git-commands] Several Git Commands

    The process of Git commands Operation 1. git commit -m 'fist post' Windows PowerShellCopyright (C) 2 ...