【计数DP】BZOJ5219 [Lydsy2017省队十连测] 最长路径

【题目】
BZOJ
给定 n n n,求一幅 n n n个点的竞赛图(所有点对之间有一条有向边),对于 i = 1 ∼ n i=1\sim n i=1n,从 1 1 1出发的最长简单路径经过点数恰好为 i i i的竞赛图个数模 P P P n ≤ 2000 , P ≤ 1 0 9 n\leq 2000,P\leq 10^9 n2000,P109

【解题思路】
对于竞赛图有一些性质:

  • 竞赛图一定有哈密尔顿路径(即沿边访问每个顶点恰好一次
  • 缩点之后按拓扑序形成一条“链”
    那么起点 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(n1

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=fni=1n1(in)gifni

初始我们有 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=1nj=1n(i1n1)(jni)gifjfnij
具体意义为枚举 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;
}
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值