淺談卡特蘭數


              淺談卡特蘭數

參考學姐的博客:http://www.cnblogs.com/yuelian/p/8719175.html

以下摘自百度百科

卡特蘭數又稱卡塔蘭數,英文名Catalan number,是組合數學中一個常出現在各種計數問題中出現的數列。以比利時的數學家歐仁·查理·卡塔蘭 (1814–1894)的名字來命名,其前幾項為 : 1, 1, 2, 5, 14, 42, 132, 429, 1430, 4862, 16796, 58786, 208012, 742900, 2674440, 9694845, 35357670, 129644790, 477638700, 1767263190, 6564120420, 24466267020, 91482563640, 343059613650, 1289904147324, 4861946401452, ...

 

卡特蘭數Cn滿足以下遞推關系  :

 

     
    
 
令h(0)=1,h(1)=1,catalan數滿足遞推式 :
  h(n)= h(0)*h(n-1)+h(1)*h(n-2) + ... + h(n-1)*h(0) (n>=2)
  例如:h(2)=h(0)*h(1)+h(1)*h(0)=1*1+1*1=2
  h(3)=h(0)*h(2)+h(1)*h(1)+h(2)*h(0)=1*2+1*1+2*1=5
另類遞推式  :
  h(n)=h(n-1)*(4*n-2)/(n+1);
遞推關系的解為:
  h(n)=C(2n,n)/(n+1) (n=0,1,2,...)
遞推關系的另類解為:
  h(n)=c(2n,n)-c(2n,n-1)(n=0,1,2,...)
 

卡特蘭數的求解方法:

    1.最基本的n^2遞推     例題:洛谷 P1044

#include<iostream>
#include<cstdio>
using namespace std;
int n;
int f[20];
int main() {
    scanf("%d", &n);
    f[0] = 1;
    f[1] = 1, f[2] = 2;
    for(int i = 3; i <= n; ++i)
        for(int j = 0; j < i; ++j)
        f[i] += f[j]*f[i-j-1];
    cout<<f[n]<<'\n';    //據說有人做過實驗cout在只輸出數字時比printf快,但如果加上換行“endl”就會慢很多
    return 0;
}
數據范圍較小,直接 n^2

    2.卡特蘭數的第n項h(n)=C(2n,n)-C(2n,n-1),所以用求組合數的方法求卡特蘭數即可,針對對一個大質數取模的代碼

#include<iostream>
#include<cstdio> 
using namespace std;
int n,p;
int js[100005];
int prime[100005];
bool vis[100005];
int cnt[100005];
int qpow(int a,int b) {    //快速冪求逆元 
    int ans = 1;
    while(b) {
        if(b&1) ans = (1ll*a*ans)%p;
        a = (1ll*a*a)%p;
        b>>=1;
    }
    return ans;
}
int main() {
    scanf("%d%d", &n, &p);//求第n項,對p取模的結果,p為大質數
    js[0] = 1;
    for(int i=1;i<=2*n;++i)
        js[i] = (1ll*js[i-1]*i)%p;
    int a = qpow(1ll*js[n]*js[n]%p,p-2), b = qpow(1ll*js[n-1]*js[n+1]%p,p-2);
    a = 1ll*a*js[2*n]%p, b = 1ll*b*js[2*n]%p;
    int ans = (a-b+p)%p; //a = C(2n,n)%p, b = C(2n,n-1)%p 
    printf("%d", ans);
    return 0;
}
View Code

    3.我們可以由第一種求法看出來卡特蘭數增長的是很快的,所以當要求的項數比較大而且不能取模時,需要用到高精,這時分解質因數求卡特蘭數就是一個很好的方法。

例題:洛谷 P2532

#include<iostream>
#include<cstdio>
using namespace std;
int n; 
int prime[1500],cnt;
bool vis[1550];
int tong[1050];
int ans[1100],len;
void cheng(int x,int sum) {    //高精乘低精
    while(sum--) {
        for(int i=1;i<=len;++i)ans[i]*=x;
        for(int i=1;i<=len;++i)
            if(ans[i]>9) {
                ans[i+1] += ans[i]/10, ans[i]%=10;
                if(i+1>len) len++;
            }
    }
}
int main()
{
    scanf("%d", &n);
    vis[0] = vis[1] = 1;
    for(int i=2;i<=2*n;++i) {    //分解質因數
        if(!vis[i])prime[++cnt]=i;
        for(int j=1;j<=cnt&&prime[j]*i<=2*n;++j) {
            vis[i*prime[j]]=1;
            if(i%prime[j]==0)break;
        }
    }
    for(int i=1;i<=cnt;++i)
    {
        int tmp=2*n;
        while(tmp/prime[i]>0) tong[i] += tmp/prime[i], tmp /= prime[i];
        tmp = n; while(tmp/prime[i]>0) tong[i]-=tmp/prime[i], tmp /= prime[i];
        tmp = n+1; while(tmp/prime[i]>0) tong[i]-=tmp/prime[i], tmp /= prime[i]; 
    }
    ans[1]=1, len=1;
    for(int i=1;i<=cnt;++i)
        if(tong[i]) cheng(prime[i], tong[i]);
    for(int i=len;i>=1;i--) printf("%d",ans[i]);
    return 0;
}
View Code

    4.卡特蘭數的遞推公式h(n)=h(n-1)*(4*n-2)/(n+1),看起來好像挺簡單的樣子,不過從來沒有用過這個遞推式     無奈

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM