CF1294F 题解
Part.0 闲话
目前题解区里大多数巨佬都是采用的树形 dp 和暴力等方法,看见没有我这种做法,欢迎指出做法问题或 hack 代码。
Part.1 题意
给定一棵树,选 \(3\) 个点 \(a, b, c\),求 \(a\) 到 \(b\) 的路径与 \(a\) 到 \(c\) 的路径与 \(b\) 到 \(c\) 的路径上一共有多少不重复的边。
Part.2 分析
我们不妨把每条边的边权设为 \(1\),这样方便我们计算贡献。
仔细思考一下后,可以发现无论如何,先把树的直径的端点选上最优,而 \(c\) 也一定是在某一个距离直径上点最远的点。
但是一个问题诞生了,怎么求这个距离最远的点呢?枚举直径上点再搜的时间复杂度是 \(O(n^2)\) 的,肯定会超时。
这时候,前文将边权设为 \(1\) 的好处就凸显出来了。
我们可以把所有直径上的边的边权全部设置为 \(0\),然后从直径的某一个端点再次搜索一遍树的直径,那么新直径的另一个端点就是我们要求的 \(c\),起点可以选在直径上的任意一点,因为此时在直径上搜不会增加答案的值。
所以答案就是旧直径的长加新直径的长。
Part.3 代码
#include <bits/stdc++.h>
using namespace std;
const int kMaxN = 2e5 + 5;
struct Node {
	int v, w, nxt;
} e[kMaxN << 1];
int n;
int dis, x;
int res, a, b, c;
int hd[kMaxN], cnt = 1;
vector<pair<int, int>> stk, d;
void add(int u, int v) {
	e[++cnt].nxt = hd[u];
	hd[u] = cnt;
	e[cnt].v = v;
	e[cnt].w = 1;
}
void dfs(int u, int fa, int s) {
	if (s >= dis) {
		dis = s;
		if (u != a && u != b && u != c) {
			x = u;
			d = stk;
		}
	}
	for (int i = hd[u]; i; i = e[i].nxt) {
		int v = e[i].v, w = e[i].w;
		if (v == fa) {
			continue;
		}
		stk.push_back({u, i});
		dfs(v, u, s + w);
		stk.pop_back();
	}
}
int main() {
	ios::sync_with_stdio(0);
	cin.tie(0), cout.tie(0);
	cin >> n;
	for (int i = 1, a, b; i < n; i++) {
		cin >> a >> b;
		add(a, b);
		add(b, a);
	}
	dis = 0;
	dfs(1, 0, 0);
	a = x;
	dis = 0;
	dfs(x, 0, 0);
	b = x;
	res += dis;
	dis = 0;
	for (auto i : d) {
		e[i.second].w = e[i.second ^ 1].w = 0;
	}
	dfs(x, 0, 0);
	c = x;
	res += dis;
	cout << res << '\n' << a << ' ' << b << ' ' << c << '\n';
	return 0;
}
值得注意的是,一定要使用链式前向星并且把双向的边全部边权赋为 \(0\)。因为我太菜了,所以只能开个栈储存直径上的边,事实上可以边搜边赋值。
时间复杂度 \(O(n)\),常数可能有一点大。
                    
                
                
            
        
浙公网安备 33010602011771号