P6591 [YsOI2020]植树

树上操作太薄弱了,根本想不出来。

思路:

首先自定义根做一遍dfs,求出sz数组,sz[i]表示i的子树大小。

在做第二遍dfs,对每一个点进行尝试,看能否为根。

当 x 点为根转移到 y 点为根时,sz发生变化,

sz[x] -= sz[y]

sz[y] += n - sz[y]

然后进行下一步搜索,回溯时改回来就好。

这道题也体现了dfs的妙处,就是回溯处理问题

#include <bits/stdc++.h>

using namespace std;
const int maxn = 2000005;
int n, sz[maxn];
vector<int>e[maxn];
priority_queue<int>q;
void dfs1(int u, int f)
{
	sz[u] = 1;
	for (int v : e[u]) {
		if (v == f) continue;
		dfs1(v, u);
		sz[u] += sz[v];
	}
}
void dfs2(int u, int f)
{
	// 思路:检验当前点能否为根;再换根搜索邻接点
	bool flg = 0;
	int t = sz[e[u][0]];
	for (int v : e[u]) {
		if (sz[v] != t) {
			flg = 1;
			break;
		}
	}
	if (!flg) {
		q.push(-u);
	}
	for (int v : e[u]) {
		if (v == f) continue;
		// change root
		sz[u] -= sz[v];
		sz[v] += sz[u];
		dfs2(v, u);
		sz[v] -= sz[u];
		sz[u] += sz[v];
	}
}
int main()
{
	scanf("%d", &n);
	if (n == 1) {
	    puts("1");
	    return 0;
	}
	for (int i = 1, a, b; i < n; i++) {
		scanf("%d%d", &a, &b);
		e[a].push_back(b);
		e[b].push_back(a);
	}
	dfs1(1, 0);
	dfs2(1, 0);
	while (q.size()) {
		printf("%d ", -q.top());
		q.pop();
	}
	printf("\n");
	return 0;
}
posted @ 2023-01-30 20:30  Vegdie  阅读(28)  评论(0编辑  收藏  举报