BZOJ_2111_[ZJOI2010]Perm 排列计数_树形DP+组合数学

Description

称一个1,2,...,N的排列P1,P2...,Pn是Magic的,当且仅当2<=i<=N时,Pi>Pi/2. 计算1,2,...N的排列中有多少是Magic的,答案可能很大,只能输出模P以后的值

Input

输入文件的第一行包含两个整数 n和p,含义如上所述。

Output

输出文件中仅包含一个整数,表示计算1,2,⋯, ���的排列中, Magic排列的个数模 p的值。

Sample Input

20 23

Sample Output

16

HINT

100%的数据中,1 ≤ ��� N ≤ 106, P��� ≤ 10^9,p是一个质数。


令fa[i]=i/2,就出现了一棵树。

f[i]表示i的子树的排列方案数。

siz[i]表示i的子树大小。

f[x]*=f[to[i]]*C(siz[x]-1,siz[to[i]])。siz[x]是不断更新的。

其中那个组合数的含义是每个儿子交错排列的方案数。

有坑点,mod可能小于n。逆元需要分类讨论。

 

代码:

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define N 1000050
typedef long long ll;
ll fac[N],inv[N],f[N];
int siz[N],n,m,mod;
int head[N],to[N<<1],nxt[N<<1],cnt;
inline void add(int u,int v) {
    to[++cnt]=v; nxt[cnt]=head[u]; head[u]=cnt;
}
ll qp(ll x,int y) {ll re=1; for(;y;y>>=1,x=x*x%mod) if(y&1) re=re*x%mod; return re;}
void init() {
    int i;
    for(fac[0]=1,i=1;i<=n;i++) fac[i]=fac[i-1]*i%mod;
    if(mod<=n) {
        inv[mod-1]=mod-1;
        for(i=mod-2;i>=0;i--) inv[i]=inv[i+1]*(i+1)%mod;
        for(i=mod;i<=n;i++) inv[i]=inv[i%mod];
    }else {
        inv[n]=qp(fac[n],mod-2);
        for(i=n-1;i>=0;i--) inv[i]=inv[i+1]*(i+1)%mod;
    }
}
ll C(int x,int y) {
    if(x<y) return 0;
    if(x<mod&&y<mod)
    return fac[x]*inv[y]%mod*inv[x-y]%mod;
    return C(x%mod,y%mod)*C(x/mod,y/mod)%mod;
}
void dfs(int x) {
    int i;
    f[x]=1;
    siz[x]=1;
    for(i=head[x];i;i=nxt[i]) {
        dfs(to[i]);
        siz[x]+=siz[to[i]];
        f[x]=f[x]*C(siz[x]-1,siz[to[i]])%mod*f[to[i]]%mod;
    }
}
int main() {
    scanf("%d%d",&n,&mod);
    init();
    int i;
    for(i=2;i<=n;i++) add(i>>1,i);
    dfs(1);
    printf("%lld\n",f[1]);
}

 

posted @ 2018-08-05 21:26  fcwww  阅读(189)  评论(0编辑  收藏  举报