poj 1095 题解(卡特兰数+递归

题目

题意:给出一个二叉树的编号,问形态。

编号依据

1:如果二叉树为空,则编号为0;

2:如果二叉树只有一个节点,则编号为1;

3:所有含有m个节点的二叉树的编号小于所有含有m+1个节点的二叉树的编号;

4:如果一棵含有m个节点的二叉树(左子树为L,右子树为R)的编号为n,要想其它含有m个节点的二叉树的编号如果大于n,则需要满足两个条件中的任意一个:1、左子树的编号大于L的左子树的编号等于L的编号,但是右子树的编号大于R的编号。(大概就是先将右子树的个数填满将变幻完后再将右子树的点向左子树转移

一道卡特兰数的题。。总的来说代码难度不高,但是思维难度还是挺高的,首先二叉树形态有几种就很想然得想到用卡特兰数啦。

递推公式

$ cat[i]=cat[i-1]* ( 4 * i-2)/(i+1) ????? cat[1]=1,cat[0]=1 $

不了解的自行百度卡特兰数

我们首先可以设一棵二叉树可能的形态种类数为 $ f[n] $ 那么对于左子数所含有的节点数 $ 0<=i<=n-1 $ 都有 $ f[n]= \sum_{i = 0}^{n-1}?f[i] * f[n-i-1] $

首先我们设当前含有x个节点的二叉树第order个排序通给定数 $ n $ 计算出当前有几个节点并且当前是第几个排序 $ order=n-Catalan(0)+Catalan(1)+...+Catalan(i-1)) ???? (Catalan(i)>=n)) $

然后我们再去考虑二叉树中左子树有多少个节点,右子树有多少节点。由题意知初始状态下,左子树为空,所有的节点均在右子树上并且所有节点只有右孩子,随着右子树的变化完后,右子树拿一个节点到左子树,然后再变化。这个过程就像时钟一样,右子树是分钟,左子树是时针,右子树的变化走满了,就开始进位,然后左子树再变化。当然这里使用的进制当然就是卡特兰数进制呀。然后只要算出左右此时的节点个数以及是第几个排序。之后层层递归下去即可。

代码

#include<cstdio>
#include<cstring>
using namespace std;
const int maxn=500000010;
int n;
long long cat[40],sum[40];
void star(){
    cat[0]=cat[1]=1;
    for(int i=2;i<35;++i){
        cat[i]=cat[i-1]*(4*i-2)/(i+1);
    }
    return;
}
void find(int x,int order){
    if(x==1){
        printf("X");
        return;
    }int i,cnt=0;
    for(i=0;cnt<order;i++){//保证左子树尽可能小
        cnt+=cat[i]*cat[x-i-1];
    }i--;
    int l=i,r=x-l-1;
    int neworder=order-(cnt-cat[l]*cat[r]);//减去其中的个数
    if(l>0){
        printf("(");
        find(l,(neworder-1)/cat[r]+1);//减一然后在后面加一相当于判断是否有余数,有余数加一
        printf(")");
    }printf("X");
    if(r>0){
        printf("(");
        find(r,(neworder-1)%cat[r]+1);//同上
        printf(")");
    }
}
int main(){
    star();
    while(scanf("%d",&n)&&n){
        int i,cnt=0;
        for(i=1;cnt<n;++i){
            cnt+=cat[i];
        }i--;
        find(i,n-(cnt-cat[i]));
        printf("\n");
    }
    return 0;
}

原文地址:https://www.cnblogs.com/donkey2603089141/p/11789209.html

时间: 2024-08-28 21:16:58

poj 1095 题解(卡特兰数+递归的相关文章

HDU-4828 卡特兰数+带模除法

题意:给定2行n列的长方形,然后把1—2*n的数字填进方格内,保证每一行,每一列都是递增序列,求有几种放置方法,对1000000007取余: 思路:本来想用组合数找规律,但是找不出来,搜题解是卡特兰数,而且还有一个难点在于N的范围是1000000,卡特兰数早已数千位,虽然有取余: 解决方法就是用在求卡特兰数的时候快速取余+带模除法: 卡特兰数递归公式1:K(n)=K(n-1) * ((4*n-2)/(n+1)); 组合数公式2:K[n] = C[2*n][n] /(n+1); 看公式1,有个除法

组合数学--卡特兰数-持续更新

参考资料: 基本介绍和各种分类 http://www.cnblogs.com/topW2W/p/5410875.html 另类递归式: h(n)=h(n-1)*(4*n-2)/(n+1);  (从n开始,更常用) 前几个卡特兰数:规定C0=1,而 分类 :  括号,栈,矩阵乘法,     凸多边形划分,二叉搜索树构造      步数上下,找零, C1=1,C2=2,C3=5,C4=14,C5=42, C6=132,C7=429,C8=1430,C9=4862,C10=16796, C11=587

POJ 2084 Game of Connections 卡特兰数

看了下大牛们的,原来这题是卡特兰数,顺便练练java.递归式子:h(0)=1,h(1)=1   h(n)= h(0)*h(n-1) + h(1)*h(n-2) + ... + h(n-1)h(0) (其中n>=2)   打表172MS import java.math.BigInteger; import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner in=new S

[NOIP2003]栈 题解(卡特兰数)

[NOIP2003]栈 Description 宁宁考虑的是这样一个问题:一个操作数序列,从1,2,一直到n(图示为1到3的情况),栈A的深度大于n. 现在可以进行两种操作: 1.将一个数,从操作数序列的头端移到栈的头端(对应数据结构栈的push操作) 2.将一个数,从栈的头端移到输出序列的尾端(对应数据结构栈的pop操作) 使用这两种操作,由一个操作数序列就可以得到一系列的输出序列; 你的程序将对给定的n,计算并输出由操作数序列1,2,-,n经过操作可能得到的输出序列的总数. Solution

[SCOI2010]生成字符串 题解(卡特兰数的扩展)

[SCOI2010]生成字符串 Description lxhgww最近接到了一个生成字符串的任务,任务需要他把n个1和m个0组成字符串,但是任务还要求在组成的字符串中,在任意的前k个字符中,1的个数不能少于0的个数.现在lxhgww想要知道满足要求的字符串共有多少个,聪明的程序员们,你们能帮助他吗? 输入格式:输入数据是一行,包括2个数字n和m; 输出格式:输出数据是一行,包括1个数字,表示满足要求的字符串数目,这个数可能会很大,只需输出这个数除以20100403的余数; Solution 1

[AHOI2012]树屋阶梯 题解(卡特兰数)

[AHOI2012]树屋阶梯 Description 暑假期间,小龙报名了一个模拟野外生存作战训练班来锻炼体魄,训练的第一个晚上,教官就给他们出了个难题.由于地上露营湿气重,必须选择在高处的树屋露营.小龙分配的树屋建立在一颗高度为N+1尺(N为正整数)的大树上,正当他发愁怎么爬上去的时候,发现旁边堆满了一些空心四方钢材(如图1.1),经过观察和测量,这些钢材截面的宽和高大小不一,但都是1尺的整数倍,教官命令队员们每人选取N个空心钢材来搭建一个总高度为N尺的阶梯来进入树屋,该阶梯每一步台阶的高度为

[HNOI2009]有趣的数列 题解(卡特兰数)

[HNOI2009]有趣的数列 Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai}: (2)所有的奇数项满足a1<a3<...<a2n-1,所有的偶数项满足a2<a4<...<a2n: (3)任意相邻的两项a2i-1与a2i(1<=i<=n)满足奇数项小于偶数项,即:a2i-1<a2i. 现在的任务是:对于给定的n,请求出有多少个不同的长度为2n的有趣的数列.

POJ 2084 Game of Connections(卡特兰数)

卡特兰数源于组合数学,ACM中比较具体的使用例子有,1括号匹配的种数.2在栈中的自然数出栈的种数.3求多边形内三角形的个数.4,n个数围城圆圈,找不相交线段的个数.5给定n个数,求组成二叉树的种数…… 此题就是第4个样例,是裸卡特兰数,但是这里牵扯的大数,可以使用java的大数类解决,但是我这里使用高精度乘法和除法模拟的(主要是java不会). 此处的递推式为H[1] = 1:H[n] = H[n-1]*(4*n-2)/(n+1){n>=2}:代码如下: 需要注意输出的形式,我这里的进制是100

【基础练习】【卡特兰数】栈 2003年NOIP全国联赛普及组第三题 题解

卡特兰数,这是一向掌握不大熟练的内容,今天借NOIP2003普及组的第三题来总结一下.当然由于原题数据弱抱,不需要高精.如果有时间我会不断补充这篇文章里的内容. 二话不说上代码 //Catalan #include<iostream> using namespace std; long long n,f[20]={0}; /*NO.1 f[n+1]=f[i]*f[n-i]from 0 to n plus f[0]=1 int main(){ cin>>n; f[0]=1;f[1]=