Codeforces 954H Path Counting(DP)

题目链接  Path Counting

题意  给定一棵高度为$n$的树,给出每一层的每个点的儿子个数(某一层的所有点儿子个数相同)。

    令$f_{k}$为长度为$k$的路径条数,求$f_{1}, f_{2}, ..., f_{2n-2}$。

 

考虑DP,设$f[i][j]$为从深度为$i$的点出发背对以$i$为根的子树(即任何时候都不进入以$i$为根的子树)走$j$步之后可以到达的点的个数。

(同一条边最多走一次)

那么$f[i][j] = f[i-1][j-1] + calc(i-1, j-1)$, $calc(i, j)$为从$i$点出发往以$i$为根的子树走$j$步能到达的点的个数。

然后这一层对答案的贡献即为$f[i][j] * g[i]$($g[i]$为第$i$层结点个数)

我们会发现这样那些一个点为另一个点的祖先的路径只统计了一次,其他路径统计了两次。

为了一致我们再求一遍前者类型路径的个数,加起来除以$2$即可。

时间复杂度$O(n^{2})$

#include <bits/stdc++.h>

using namespace std;

#define rep(i, a, b)	for (int i(a); i <= (b); ++i)
#define dec(i, a, b)	for (int i(a); i >= (b); --i)

const int N = 5e3 + 10;
const int mod = 1e9 + 7;
const int rev = 5e8 + 4;

int a[N << 1], f[N][N << 1];
int num[N << 1], c[N << 2], s[N << 1], inv[N << 1];
int ans[N << 1];
int n;

inline int Pow(int a, int b, int mod){
        int ret = 1;
        for (; b; b >>= 1, a = 1ll * a * a % mod) if (b & 1) ret = 1ll * ret * a % mod;
        return ret;
}


int calc(int x, int y){
	int ret = (a[x] + mod - 1) % mod;
	ret = 1ll * ret * c[x + y - 1] % mod;
	ret = 1ll * ret * inv[x] % mod;
	return ret;
}


int main(){

	scanf("%d", &n);
	rep(i, 1, n - 1) scanf("%d", a + i);

	num[1] = 1; 
	rep(i, 2, n) num[i] = 1ll * num[i - 1] * a[i - 1] % mod;

	c[0] = 1;
	rep(i, 1, n) c[i] = 1ll * c[i - 1] * a[i] % mod;

	rep(i, 1, n) inv[i] = Pow(c[i], mod - 2, mod);

	s[0] = 0;
	rep(i, 1, n) s[i] = (s[i - 1] + num[i]) % mod;

	rep(i, 2, n){
		f[i][1] = 1;
		rep(j, 2, 2 * n - 2){
			f[i][j] = (1ll * f[i - 1][j - 1] + calc(i - 1, j - 1)) % mod;
		}
	}

	rep(i, 1, 2 * n - 2){
		rep(j, 1, n){
			ans[i] = (1ll * ans[i] + 1ll * f[j][i] * num[j]) % mod;
		}
	}

	rep(i, 1, n - 1){
		ans[i] = (ans[i] + s[n]) % mod;
		ans[i] = (ans[i] - s[i] + mod) % mod;
	}

	rep(i, 1, 2 * n - 2) ans[i] = 1ll * ans[i] * rev % mod;
	rep(i, 1, 2 * n - 2) printf("%d\n", ans[i]);
	return 0;
}

 

posted @ 2018-03-28 13:10  cxhscst2  阅读(261)  评论(0编辑  收藏  举报