SGU 130 Circle(卡特兰数)

Description
一个圆上顺次有2k个点,在这2k个点上连k条弦(每个点只能连一条边),问最少可以将这个圆分为几部分,有多少种方法
Input
一个整数k(1<=k<=30)
Output
两个整数表示这个圆有多少种方法被被分为最少的几部分
Sample Input
2
Sample Output
2 3
Solution
显然这个圆最少会被分成k+1部分,即2k个点两两连互不相交的k条边
令dp[n]为在2n个点中连n条边把一个简单图形分成n+1部分的方法数那么一个圆可以通过第1个点与第2,4,…,2k个点连接的弦被分成两个半圆,且两边都是偶数个点,不妨将1和2i这两个点放入其中一个半圆中(不能与奇数点相连是因为此时两边都是奇数个点,两边必然都剩一个点,这两个点相连会和第一个点这条弦相交),则有转移方程
dp[n]=dp[1]*dp[n-1]+dp[2]*dp[n-2]+…+dp[n-1]*dp[1]
可以看出这是卡特兰数的递推公式,这里用另一个卡特兰数的递推公式
ktl[n]=(4*n-2)*ktl[n-1]/(n+1) 会更快
Code

#include<cstdio>
#include<iostream>
using namespace std;
typedef long long ll;
int k;
ll ktl[33];
void init()
{
    ktl[1]=1ll;
    for(int i=2;i<=30;i++)
        ktl[i]=1ll*(4*i-2)*ktl[i-1]/(i+1); 
}
int main()
{
    init();
    while(~scanf("%d",&k))
        printf("%lld %d\n",ktl[k],k+1);
}

你可能感兴趣的:(SGU 130 Circle(卡特兰数))