首页 > 其他分享 >【luogu CF1707D】Partial Virtual Trees(容斥)(DP)

【luogu CF1707D】Partial Virtual Trees(容斥)(DP)

时间:2023-01-14 02:00:10浏览次数:70  
标签:子树 Partial int luogu mo 容斥 子集 mul return

Partial Virtual Trees

题目链接:luogu CF1707D

题目大意

给你一棵以 1 为根的数,问你对于每个长度,有多少个点集序列,第一个点集是全部点,最后一个点集只有 1 号点,且中间每个点集都是上一个点集的真子集,而且每个点集内两两点的 LCA 都在点集中。

思路

发现其实这个每次是真子集不太好处理,我们考虑如果不是真子集,就是子集。
那如果求出来了,我们不难通过容斥得到真子集的答案。
\(f_i=g_i-\sum\limits_{j=1}^{i-1}f_{i-1}\binom{i}{j}\)
(就是选那几次操作删的点集是空的)

那考虑子集怎么弄,考虑删的过程,要怎么删才能满足是虚树。
那考虑树上的 DP,删一个子树的根会在什么时候,有两种情况:
一是最后删,二是先删剩一个子树,然后把它删了,再删剩下的子树。
那你可以设 \(f_{i,j}\) 为 \(j\) 步删完 \(i\) 的子树的方案数。
然后不难首先最后删其实就是 \(\prod\limits_{son}(\sum\limits_{k=1}^{j}f_{son,k})\)
那这个用前缀和维护一个 \(g_{i,j}=\sum\limits_{k=1}^jf_{i,k}\) 即可求。

那第二个的话你枚举剩下的子树,那其他子树的方案数就是也是上面那些乘起来(不能乘你剩下的那个),不过要注意的是这些只能是不超过 \(j-1\) 次就删完。
那你就把前面那个东西也前缀和一下(先枚举剩下的子树,再枚举删的次数),跟 \(f_{sp_{son},j}\) 乘起来就可以共吸纳给 \(f_{i,j}\) 啦。
然后我们给容斥用的就是 \(f_{1,i}\)。

不过要注意的一点是第二个删法在当前根是 \(1\) 的时候不能用,因为你最后要留下的是 \(1\),换而言之 \(1\) 是要最晚删去的。

代码

#include<cstdio>

using namespace std;

const int N = 2e3 + 100;
struct node {
	int to, nxt;
}e[N << 1];
int n, mo, le[N], KK, sz[N], tmp[N];
int jc[N], inv[N], invs[N], f[N][N], g[N][N];
int sum[N], s0[N], ans[N];

int add(int x, int y) {return x + y >= mo ? x + y - mo : x + y;}
int dec(int x, int y) {return x < y ? x - y + mo : x - y;}
int mul(int x, int y) {return 1ll * x * y % mo;}
int C(int n, int m) {
	if (n < 0 || m < 0 || n < m) return 0;
	return mul(jc[n], mul(invs[m], invs[n - m]));
}
int ksm(int x, int y) {
	int re = 1;
	while (y) {
		if (y & 1) re = mul(re, x);
		x = mul(x, x); y >>= 1;
	}
	return re;
}
int Inv(int x) {return ksm(x, mo - 2);}

void Add(int x, int y) {
	e[++KK] = (node){y, le[x]}; le[x] = KK;
}

void work(int now, int father) {
	for (int i = le[now]; i; i = e[i].nxt)
		if (e[i].to != father)
			work(e[i].to, now);
	for (int i = 1; i <= n; i++) {
		sum[i] = 1; s0[i] = 0;
		for (int j = le[now]; j; j = e[j].nxt)
			if (e[j].to != father) {
				if (!g[e[j].to][i]) s0[i]++;
					else sum[i] = mul(sum[i], g[e[j].to][i]);
			}
	}
	for (int i = 1; i <= n; i++)
		if (s0[i]) f[now][i] = 0;
			else f[now][i] = sum[i];
	if (now != 1) {
		for (int i = le[now]; i; i = e[i].nxt)
			if (e[i].to != father) {
				for (int j = 1; j <= n; j++) {
					if (!g[e[i].to][j]) s0[j]--;
						else sum[j] = mul(sum[j], Inv(g[e[i].to][j]));
					if (!s0[j]) tmp[j] = add(tmp[j - 1], sum[j]);
						else tmp[j] = tmp[j - 1];
					f[now][j] = add(f[now][j], mul(f[e[i].to][j], tmp[j - 1]));
					if (!g[e[i].to][j]) s0[j]++;
						else sum[j] = mul(sum[j], g[e[i].to][j]);
				}
			}
	}
	for (int i = 1; i <= n; i++) g[now][i] = add(g[now][i - 1], f[now][i]);
}

int main() {
	scanf("%d %d", &n, &mo);
	for (int i = 1; i < n; i++) {
		int x, y; scanf("%d %d", &x, &y);
		Add(x, y); Add(y, x);
	}
	
	jc[0] = 1; for (int i = 1; i < N; i++) jc[i] = mul(jc[i - 1], i);
	inv[0] = inv[1] = 1; for (int i = 2; i < N; i++) inv[i] = mul(inv[mo % i], mo - mo / i);
	invs[0] = 1; for (int i = 1; i < N; i++) invs[i] = mul(invs[i - 1], inv[i]);
	
	work(1, 0);
	for (int i = 1; i < n; i++) {
		ans[i] = f[1][i];
		for (int j = 1; j < i; j++)
			ans[i] = dec(ans[i], mul(ans[j], C(i, j)));
		printf("%d ", ans[i]);
	}
	
	return 0;
}

标签:子树,Partial,int,luogu,mo,容斥,子集,mul,return
From: https://www.cnblogs.com/Sakura-TJH/p/luogu_CF1707D.html

相关文章

  • luogu P3518 [POI2011]SEJ-Strongbox | loj #2160. 「POI2011 R2 Day0」保险箱 Strong
    代码已在loj上不开O2通过。下文均在\(Z_n\)下考虑。首先,你考虑选出一些数,能组成的数。即ttps://www.cnblogs.com/xugangfan/p/17040634.html那么对于一个不在群......
  • 类型“AxiosHeaders | Partial<RawAxiosHeaders & MethodsHeaders & CommonHeaders>”
    背景使用TS封装axios,在请求拦截器里添加Authorization携带Token:config.headers.Authorization=`Bearer${token}`TS报错:axios版本:1.2.2解决1.2.2版本的AxiosHea......
  • Luogu P5465 [PKUSC2018] 星际穿越
    观察可以发现一个结论,可以视作每个点\(i\)可以一步到达\(l_i\simn\)的每一个点。发现对于\(a<b<x\),\(dist(a,x)\gedist(b,x)\)第一步是相当特殊的,因为第一步......
  • CF1768E Partial Sorting
    可能更好的阅读体验题目传送门题目翻译题目解析显然我们可以证明\(f(p)\in\{0,1,2,3\}\)\(f(p)=0\)显然只有\(s_1=1\)种。考虑\(f(p)=1\)如果前面交换一次,那么......
  • luogu P5291 [十二省联考 2019] 希望
    题面传送门真的很想吐。题目的意思大概就是在一棵树上选出\(k\)个联通块,使得这\(k\)个联通块有交。显然联通块的交还是联通块,因此转化为对联通块计数。而联通块个数等于......
  • CF1768E Partial Sorting - 组合数学 -
    题目链接:https://codeforces.com/contest/1768/problem/E题解:记P1为将\(1..2\timesn\)排序,P2为将\(n+1..3\timesn\)排序首先观察到答案一定不会超过3(P1P2......
  • [luogu]P2123 皇后游戏
    题目传送门分析和国王游戏一样的思路直接考虑邻项交换观察易知排在后面的大臣获得的奖赏一定更多假设前\(i-1\)位左手上的数和为\(a\),第\(i-1\)位获得奖赏为\(......
  • luogu P1971 [NOI2011] 兔兔与蛋蛋游戏
    题面传送门没想到二分图博弈这么早就考了?首先我们发现,如果将和起点行列之和奇偶性一样的点设为黑色,其余设为白色,那么每次空格只会在异色边之间走,而不会在同色边之间走。......
  • Luogu P4592 [TJOI2018]异或 做题记录
    随机跳的。树上维护序列,显然树剖。维护异或,显然01trie。01trie维护区间异或,显然可持久化一下。看到时限很大,显然可以双log。于是跑一边树剖,再根据id暴力建一个可......
  • luogu P5037 抓捕
    ##题意有$n$个点的图,任意一点$x$到可去另外一点$y$必须互质,即$\gcd(x,y)=1$。图无边权,但是拥有点权。求到终点$en$的最短距离。---##思路此题需使用Dijks......