1. 程式人生 > >bzoj2111: [ZJOI2010]Perm 排列計數

bzoj2111: [ZJOI2010]Perm 排列計數

一定的 https 字典 line cst www. getch inline http

題目鏈接

bzoj2111: [ZJOI2010]Perm 排列計數

題解

序列大小關系構成樹形小根堆關系
設f[i]表示大小為i的堆由多少種形態
那麽f[n] = f[l] * f[r] * C(n - 1,l),l,r為左右子數大小
對於每個n左子樹的大小是一定的,可以處理出
組合數取膜要lucas
如果是求字典序最小就變成九省聯考題了2333

代碼

#include<cstdio> 
#include<algorithm> 
inline int read() { 
    int x = 0,f = 1; 
    char c = getchar(); 
    while(c < '0' || c > '9') c = getchar(); 
    while(c <= '9' && c >= '0') x = x * 10 + c - '0',c = getchar(); 
    return x * f; 
} 
#define int long long 
#define LL long long 
const int maxn = 1000007; 
int f[maxn],ls[maxn]; 
int p,n; 
int jc[maxn];  
int isl[maxn]; 
inline LL inv(LL x) { 
    int k = p - 2; 
    int ret = 1; 
    for(;k;k >>= 1,x = 1ll * x * x % p) 
        if(k & 1) ret = 1ll * ret * x % p; 
    return ret; 
} 
int C (int x,int y) { 
    if(x < y) return 0; 
    if(x < p) return 1ll * jc[x] * inv(jc[y]) % p * inv(jc[x - y]) % p; 
    return 1ll * C(x / p,y / p) * C(x % p,y % p) % p; 
} 
int F(int x) { 
    if(f[x]) return f[x]; 
    return 1ll * F(ls[x]) * F(x - 1 - ls[x]) % p * C(x - 1,ls[x]) % p; 
} 
main() { 
    n = read(); p = read(); 
    f[1] = f[2] = 1; 
    isl[2] = 1; 
    for(int i = 3;i <= n;++ i) isl[i] = isl[i >> 1]; 
    for(int i = 2;i <= n;++ i) ls[i] = ls[i - 1] + isl[i]; 
    jc[0] = 1; 
    for(int i = 1;i <= n;++ i) jc[i] = jc[i - 1] * i % p; 
    printf("%lld\n",F(n)); 
    return 0; 
} 
?

bzoj2111: [ZJOI2010]Perm 排列計數