【题目】
BZOJ
给定
n
n
n,求一幅
n
n
n个点的竞赛图(所有点对之间有一条有向边),对于
i
=
1
∼
n
i=1\sim n
i=1∼n,从
1
1
1出发的最长简单路径经过点数恰好为
i
i
i的竞赛图个数模
P
P
P。
n
≤
2000
,
P
≤
1
0
9
n\leq 2000,P\leq 10^9
n≤2000,P≤109
【解题思路】
对于竞赛图有一些性质:
- 竞赛图一定有哈密尔顿路径(即沿边访问每个顶点恰好一次
- 缩点之后按拓扑序形成一条“链”
那么起点 1 1 1一定在这条路径上,且从 1 1 1出发最长路径上的点数等于 1 1 1所在 s c c scc scc点数 + + +拓扑序在 1 1 1以后的 s c c scc scc的点的总数。
令 f n f_n fn表示 n n n个节点竞赛图个数,那么 f n = 2 n ( n − 1 2 f_n=2^{\frac {n(n-1} 2} fn=22n(n−1
令 g n g_n gn表示有 n n n个节点的竞赛图 s c c scc scc,我们枚举拓扑序最小的 s c c scc scc进行容斥,那么有: g n = f n − ∑ i = 1 n − 1 ( n i ) g i f n − i g_n=f_n-\sum\limits_{i=1}^{n-1}{n\choose i}g_if_{n-i} gn=fn−i=1∑n−1(in)gifn−i
初始我们有 f 0 = 1 , g 1 = 1 f_0=1,g_1=1 f0=1,g1=1,然后递推即可。
那么我们枚举
1
1
1所在最后有答按,令
a
n
s
k
ans_k
ansk表示从
1
1
1出发最长路径为
k
k
k的方案数
a
n
s
k
=
∑
i
=
1
n
∑
j
=
1
n
(
n
−
1
i
−
1
)
(
n
−
i
j
)
g
i
f
j
f
n
−
i
−
j
ans_k=\sum_{i=1}^n\sum_{j=1}^n {n-1\choose i-1}{n-i\choose j}g_if_jf_{n-i-j}
ansk=i=1∑nj=1∑n(i−1n−1)(jn−i)gifjfn−i−j
具体意义为枚举
1
1
1中
s
c
c
scc
scc点数以及拓扑序在
1
1
1后面的
s
c
c
scc
scc总点数。
预处理组合数可以得到一个 O ( n 2 ) O(n^2) O(n2)的做法。
【参考代码】
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=2005;
int n,mod,c[N][N],f[N],g[N],ans[N];
int upm(int x){return x>=mod?x-mod:(x<0)?x+mod:x;}
void up(int &x,int y){x=upm(x+y);}
int mul(int x,int y){return 1ll*x*y%mod;}
int qpow(int x,int y){int res=1;for(;y;y>>=1,x=mul(x,x))if(y&1)res=mul(res,x);return res;}
int main()
{
#ifndef ONLINE_JUDGE
freopen("BZOJ5219.in","r",stdin);
freopen("BZOJ5219.out","w",stdout);
#endif
scanf("%d%d",&n,&mod);
for(int i=0;i<=n;++i)
{
c[i][0]=c[i][i]=1;
for(int j=1;j<i;++j) c[i][j]=upm(c[i-1][j]+c[i-1][j-1]);
}
for(int i=0;i<=n;++i) f[i]=qpow(2,i*(i-1)/2);
for(int i=1;i<=n;++i)
{
g[i]=f[i];
for(int j=1;j<i;++j) up(g[i],-mul(c[i][j],mul(g[j],f[i-j])));
}
for(int i=1;i<=n;++i) for(int j=0;j<=n-i;++j)
up(ans[i+j],1ll*mul(c[n-1][i-1],c[n-i][j])*mul(g[i],f[j])%mod*f[n-i-j]%mod);
for(int i=1;i<=n;++i) printf("%d\n",ans[i]);
return 0;
}