2023年浙大城市学院新生程序设计竞赛(同步赛)G

登录---专业IT笔试面试备考平台_牛客网

题意

思路

首先想法非常单一,一定是去枚举操作点,然后看它染白和不染的价值差值

也就是说,把一个黑色结点染白之后,对哪些结点的价值会影响

不难想象其实就是操作结点的子树和该点连通的黑色连通块的所有结点,对这些结点会有影响

那么差值其实就是黑色连通块大小 * 操作点到最近的白色祖先的距离

黑色连通块容易用树形dp求,然后就是操作点到最近白色祖先距离怎么求

dfs即可,记录上次的白色结点是什么即可

cpp 复制代码
#include <bits/stdc++.h>
 
#define int long long
 
constexpr int N = 2e5 + 10;
 
std::vector<int> adj[N];
 
int n;
int a[N];
int lst[N];
int sz[N], dep[N], F[N][33];
int dp[N];
 
void dfs(int u, int fa) {
	sz[u] = 1;
	dep[u] = dep[fa] + 1;
	F[u][0] = fa;
	for (int i = 1; i <= 30; i++) {
		F[u][i] = F[F[u][i - 1]][i - 1];
	}
	for (auto v : adj[u]) {
		if (v == fa) continue;
		dfs(v, u);
		sz[u] += sz[v];
	}
}
void dfs2(int u, int fa) {
	if (a[u] == 1) dp[u] = 1;
	for (auto v : adj[u]) {
		if (v == fa) continue;
		dfs2(v, u);
		if (a[v] == 1) {
			dp[u] += dp[v];
		}
	}
}
void dfs3(int u, int fa, int last) {
	lst[u] = last;
	for (auto v : adj[u]) {
		if (v == fa) continue;
		int clst = 0;
		if (a[v]) {
			clst = last;
		}else {
			clst = v;
		}
		dfs3(v, u, clst); 
	}
}
int lca(int u, int v) {
	if (dep[u] < dep[v]) {
		std::swap(u, v);
	}
	for (int j = 30; j >= 0; j --) {
		if (dep[F[u][j]] >= dep[v]) {
			u = F[u][j];
		} 
	}
	if (u == v) return u;
	for (int j = 30; j >= 0; j --) {
		if (F[u][j] != F[v][j]) {
			u = F[u][j];
			v = F[v][j];
		}
	}
	return F[u][0];
}
int dis(int u, int v) {
	return dep[u] + dep[v] - 2 * dep[lca(u, v)];
}
 
void solve() {
	std::cin >> n;
	for (int i = 1; i <= n; i ++) {
		std::cin >> a[i];
	}
	for (int i = 1; i <= n - 1; i ++) {
		int u, v;
		std::cin >> u >> v;
		adj[u].push_back(v);
		adj[v].push_back(u);
	}
	dfs(1, 0);
	dfs3(1, 0, 1);
	for (int i = 1; i <= n; i ++) {
		lst[i] = dis(i, lst[i]);
	}
	dfs2(1, 0);
	int cur = 0;
	for (int u = 1; u <= n; u ++) {
		if (a[u] == 1) {
			cur += lst[u];
		}
	}
	int ans = cur;
	for (int u = 1; u <= n; u ++) {
		if (!a[u]) continue;
		int res = cur;
		res -= dp[u] * lst[u];
		ans = std::min(ans, res);
	}
	std::cout << ans << "\n";
}
signed main() {
	std::ios::sync_with_stdio(false);
	std::cin.tie(nullptr);
	
	int t = 1;
	while(t --) {
		solve();
	}
	return 0;
} 
相关推荐
D_C_tyu2 分钟前
HTML | 结合Canvas开发具有智能寻路功能的贪吃蛇小游戏实战详解
javascript·算法·游戏·html·bfs
小王不爱笑1327 分钟前
HashMap 扩容全流程
java·数据结构·算法
代码探秘者9 分钟前
【算法篇】4.前缀和
java·数据库·后端·python·算法·spring
计算机安禾9 分钟前
【数据结构与算法】第4篇:算法效率衡量:时间复杂度和空间复杂度
java·c语言·开发语言·数据结构·c++·算法·visual studio
Oueii15 分钟前
嵌入式LinuxC++开发
开发语言·c++·算法
sw12138916 分钟前
嵌入式C++驱动开发
开发语言·c++·算法
2501_9249526917 分钟前
C++中的适配器模式
开发语言·c++·算法
智驱力人工智能23 分钟前
馆藏文物预防性保护依赖的图像分析技术 文物损害检测 文物破损检测 文物损害识别误报率优化方案 文物安全巡查AI系统案例 智慧文保AI监测
人工智能·算法·安全·yolo·边缘计算
wuguan_23 分钟前
Halcon中值滤波,均值滤波,高斯滤波
算法·halcon
☆56624 分钟前
C++安全编程指南
开发语言·c++·算法