BZOJ 3732: Network

题目链接:传送门

kruskal重构树
板子到不想写…

那么这是个什么东西呢?
构造很简单
我们在构造生成树的时候
每遇到两个不在同一集合的节点
就新建一个节点并将这个节点设为那两个节点的父亲
将新建节点的权值设为原来两个节点之间的边权
这样就将边权转化成了点权
由于我们的边权是有序的
所以重构出的树的根节点的权值一定是最大的或者是最小的
可以举一个例子自己构造试一下
比如:
1 3 2
3 4 4
1 2 5
2 4 6
2 3 11
构造出来应该是这个样子的
在这里插入图片描述
567就是我们新建出来的节点
点权可以自己画上试试
这样两个点的lca的点权就是两点路径边权最大的最小值
因为边权越大的边构成的点越靠上
所以这样走的是最小的边权
代码也很好写

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <complex>
#include <algorithm>
#include <climits>
#include <queue>
#include <map>
#include <set>
#include <vector>
#include <iomanip>
#define A 1000010
#define B 2010

using namespace std;
typedef long long ll;
struct node {
	int next, to;
}e[A];
int head[A], num;
void add(int fr, int to) {
	e[++num].next = head[fr];
	e[num].to = to;
	head[fr] = num;
}
struct Edge {
	int a, b, c;
	friend bool operator < (const Edge a, const Edge b) {
		return a.c < b.c;
	}
}p[A];
int n, m, k, w[A], cnt, fa[A], a, b;
int find(int x) {
	if (fa[x] == x) return x;
	else return fa[x] = find(fa[x]);
}
int siz[A], faa[A], dep[A], son[A], dfn[A], ctt, top[A];
void prepare(int fr) {
	siz[fr] = 1;
	for (int i = head[fr]; i; i = e[i].next) {
		int ca = e[i].to;
		if (ca == faa[fr]) continue;
		faa[ca] = fr;
		dep[ca] = dep[fr] + 1;
		prepare(ca);
		siz[fr] += siz[ca];
		if (siz[ca] > siz[son[fr]]) son[fr] = ca;
	}
}
void dfs(int fr, int tp) {
	dfn[fr] = ++ctt, top[fr] = tp;
	if (son[fr]) dfs(son[fr], tp);
	for (int i = head[fr]; i; i = e[i].next) {
		int ca = e[i].to;
		if (ca == faa[fr] or ca == son[fr]) continue;
		dfs(ca, ca);
	}
}
int lca(int x, int y) {
	while (top[x] != top[y]) {
		if (dep[top[x]] < dep[top[y]]) swap(x, y);
		x = faa[top[x]];
	}
	return dep[x] < dep[y] ? w[x] : w[y];
}

int main(int argc, char const *argv[]) {
	cin >> n >> m >> k;
	for (int i = 1; i <= m; i++) scanf("%d%d%d", &p[i].a, &p[i].b, &p[i].c);
	sort(p + 1, p + m + 1); cnt = n;
	for (int i = 1; i <= n; i++) fa[i] = i;
	for (int i = 1; i <= m; i++) {
		int fx = find(p[i].a), fy = find(p[i].b);
		if (fx == fy) continue;
		w[++cnt] = p[i].c;
		fa[cnt] = fa[fx] = fa[fy] = cnt;
		add(cnt, fx); add(fx, cnt);
		add(cnt, fy); add(fy, cnt);
	}
	prepare(cnt); dfs(cnt, 0);
	while (k--) {
		scanf("%d%d", &a, &b);
		printf("%d\n", lca(a, b));
	}
}
  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

良月澪二

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值