【APIO2020】交换城市(Kruskal重构树)
Description
给定一个 \(n\) 个点,\(m\) 条边的无向连通图,边带权。
\(q\) 次询问,每次询问两个点 \(x, y\),求两点间的次小瓶颈路。不存在输出 -1。
Hint
- \(1\le n\le 10^5\)
- \(n-1\le m\le 2\times 10^5\)
- \(1\le q\le 2\times 10^5\)
Solution
若是求最小瓶颈路的话我们会有一个这样的思路:
由于 最小瓶颈路必定在这个图的最小生成树(MST)上,因此我们可以建出 Kruskal 重构树。
何为 Kruskal 重构树?我们回忆一下 Kruskal 求最小生成树的算法,是选取一个跨越两个连通块的边 \((u, v)\) 进行连接。但现在我们不连边,而书 新建一个结点 \(x\),从 \(x\) 向 \(u, v\) 各连一条边。每一个新建的结点都带有一个 点权,权值大小即为对应边的边权。最终得到的 \(2n-1\) 个结点的树形结构,即为 Kruskal 重构树。
如果我们深入挖掘这棵树的性质,不难发现这些新建点必然对应着 MST 中的边。于是 \(x\) 到 \(y\) 的瓶颈路一定是重构树上的一条路径。而我们又可以看出,祖先结点的加入时间是晚于子孙的,因此 祖先的点权必然大于其子孙。最后,不难得到结论——\(x, y\) 两点的瓶颈路即为两点在重构树上的 \(LCA\) 的点权。
以上就是最小瓶颈路的求法。
对于次小瓶颈路我们可以稍加拓展。
既为『次小』,那么必然是有两条。
注意一下这里,我们需要对上述的建树方法做一点改动,原本发现是同一个连通块是我们会直接舍弃这条边。然而我们求次小的,这条边并不一定毫无贡献。直接往所在的连通块连接即可。
由于说明需要,这个重构树还有一个比较显然的性质——一个重构树上的子树,若树根 \(x\) 点权为 \(w\),那么这棵子树对应一个连通块,块内所有点可以通过权值不超过 \(w\) 的边可以互相到达。
于是对于两个询问点 \(x, y\),我们只要找到重构树上这样一个点 \(z\),满足 \(z\) 是两点的祖先,并且其对应连通块存在次小瓶颈路。在建重构树的过程中,我们通过一个标记 \(type\) 来表示子树所对应的连通块是否存在。
对于一个链状的连通块,很显然只有一条路径。
根据这一点,我们可以设计出如下判断条件:
- 若新建点 \(x\) 只连向了一个点,说明该连通块中 存在环 了,那么 \(type = 1\)。
- 若连向了两个连通块:
- 如果 两个连通块存在一个非链状块,那么合并后显然也是,\(type = 1\)。
- 如果加上这条边后,发现一个 度数超过 \(2\) 的顶点,那么同样为非链块,\(type = 1\)。
- 否则 \(type = 0\)。
处理询问的时候,我们先其求出 \(LCA\),然后 倍增 地往上跳祖先,直到找到一个 深度最大 \(type = 1\) 的点。其点权即为答案。
这个算法的时间复杂度为 \(O((n+m+q)\log (n+m))\)。
/*
* Author : _Wallace_
* Source : https://www.cnblogs.com/-Wallace-/
* Problem : APIO2020 交换城市
*/
#include "swap.h"
#include <algorithm>
#include <vector>
using namespace std;
const int M = 2e5 + 5;
const int N = 1e5 + 5;
int n, m;
struct Edge {
int u, v, w;
} e[M];
const int V = N + M;
const int LogV = 20;
vector<int> tr[V];
int fa[V][LogV];
int dep[V];
int val[V], ind[V];
bool type[V];
int tot = 0;
int uset[V];
int find(int x) {
return x == uset[x] ? x : uset[x] = find(uset[x]);
}
void caldep(int x) {
dep[x] = dep[fa[x][0]] + 1;
for (auto y : tr[x]) caldep(y);
}
void Kruskal() {
for (int i = 1; i <= n; i++) uset[i] = i;
sort(e + 1, e + m + 1, [&](Edge a, Edge b) {
return a.w < b.w;
});
tot = n;
for (int i = 1; i <= m; i++) {
int u = e[i].u, fu = find(u);
int v = e[i].v, fv = find(v);
int w = e[i].w;
++tot;
if (fu == fv) {
val[tot] = w;
fa[fu][0] = tot;
uset[fu] = uset[tot] = tot;
type[tot] = true;
tr[tot].push_back(fu);
} else {
val[tot] = w;
fa[fu][0] = fa[fv][0] = tot;
uset[fu] = uset[fv] = uset[tot] = tot;
if (++ind[u] > 2 || ++ind[v] > 2) type[tot] = true;
if (type[fu] || type[fv]) type[tot] = true;
tr[tot].push_back(fu);
tr[tot].push_back(fv);
}
}
type[0] = 1;
for (int j = 1; j < LogV; j++)
for (int i = 1; i <= tot; i++)
fa[i][j] = fa[fa[i][j - 1]][j - 1];
dep[0] = 0;
caldep(tot);
}
int lca(int x, int y) {
if (dep[x] < dep[y]) swap(x, y);
for (int i = LogV - 1; ~i; --i)
if (dep[x] - (1 << i) >= dep[y])
x = fa[x][i];
if (x == y) return x;
for (int i = LogV - 1; ~i; --i)
if (fa[x][i] != fa[y][i])
x = fa[x][i], y = fa[y][i];
return fa[x][0];
}
void init(int tmp_n, int tmp_m, vector<int> U, vector<int> V, vector<int> W) {
n = tmp_n, m = tmp_m;
for (int i = 0; i < m; i++)
e[i + 1] = Edge{U[i] + 1, V[i] + 1, W[i]};
Kruskal();
}
int getMinimumFuelCapacity(int x, int y) {
int z = lca(++x, ++y);
if (type[z]) return val[z];
for (int j = LogV - 1; ~j; --j)
if (!type[fa[z][j]])
z = fa[z][j];
z = fa[z][0];
return z ? val[z] : -1;
}
【APIO2020】交换城市(Kruskal重构树)的更多相关文章
- BZOJ5415[Noi2018]归程——kruskal重构树+倍增+堆优化dijkstra
题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 n 个节点.m 条边的无向连通图(节点的编号从 1 至 n).我们依次用 l,a 描述一条边的长度.海 ...
- [SCOI2013]摩托车交易 kruskal重构树(最大生成树) 倍增
---题面--- 题解: 这题想法简单,,,写起来真的是失智,找了几个小时的错误结果是inf没开到LL范围.... 首先我们需要找到任意两点之间能够携带黄金的上限值,因为是在经过的道路权值中取min, ...
- NOI2018归程(Kruskal重构树)
题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 n 个节点.m 条边的无向连通图(节点的编号从 1 至 n). 我们依次用 l,a 描述一条边的长度. ...
- Luogu P4768 [NOI2018]归程(Dijkstra+Kruskal重构树)
P4768 [NOI2018]归程 题面 题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 \(n\) 个节点. \(m\) 条边的无向连通图(节点的编 ...
- Luogu P1967 货车运输(Kruskal重构树)
P1967 货车运输 题面 题目描述 \(A\) 国有 \(n\) 座城市,编号从 \(1\) 到 \(n\) ,城市之间有 \(m\) 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 \ ...
- NOIP 2013 提高组 洛谷P1967 货车运输 (Kruskal重构树)
题目: A 国有 nn 座城市,编号从 11 到 nn,城市之间有 mm 条双向道路.每一条道路对车辆都有重量限制,简称限重. 现在有 qq 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情 ...
- [bzoj 3732] Network (Kruskal重构树)
kruskal重构树 Description 给你N个点的无向图 (1 <= N <= 15,000),记为:1-N. 图中有M条边 (1 <= M <= 30,000) ,第 ...
- 【BZOJ 3732】 Network Kruskal重构树+倍增LCA
Kruskal重构树裸题, Sunshine互测的A题就是Kruskal重构树,我通过互测了解到了这个神奇的东西... 理解起来应该没什么难度吧,但是我的Peaks连WA,,, 省选估计要滚粗了TwT ...
- 【BZOJ-3545&3551】Peaks&加强版 Kruskal重构树 + 主席树 + DFS序 + 倍增
3545: [ONTAK2010]Peaks Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1202 Solved: 321[Submit][Sta ...
随机推荐
- shell编程之算术扩展(引号、命令替换、算术扩展)
1.单引号 .双引号.反引号的区别 单引号:忽略所有特殊字符 双引号:忽略大部分特殊字符($ `等字符除外) [root@tlinux shell]# echo '*' * [root@tlinux ...
- linux下内存释放
细心的朋友会注意到,当你在linux下频繁存取文件后,物理内存会很快被用光,当程序结束后,内存不会被正常释放,而是一直作为caching.这个问题,貌似有不少人在问,不过都没有看到有什么很好解决的办法 ...
- JVM简单入门
目录 初识JVM 双亲委派机制 沙箱安全机制 Native PC计数器 方法区 栈 堆 工具分析OOM GC算法 GC算法总结 JMM 初识JVM JVM的位置:jre中包含jvm. 双亲委派机制 双 ...
- [原题复现][CISCN 2019 初赛]WEB-Love Math(无参数RCE)[未完结]
简介 原题复现: 考察知识点:无参数命令执行 线上平台:https://buuoj.cn(北京联合大学公开的CTF平台) 榆林学院内可使用信安协会内部的CTF训练平台找到此题 源码审计 代码 1 ...
- JUC并发工具包之CyclicBarrier & CountDownLatch的异同
1.介绍 本文我们将比较一下CyclicBarrier和CountDownLatch并了解两者的相似与不同. 2.两者是什么 当谈到并发,将这两者概念化的去解释两者是做什么的,这其实是一件很有挑战的事 ...
- 最全总结 | 聊聊 Python 办公自动化之 Word(下)
1. 前言 关于 Word 文档的读写,前面两篇文章分别进行了一次全面的总结 最全总结 | 聊聊 Python 办公自动化之 Word(上) 最全总结 | 聊聊 Python 办公自动化之 Word( ...
- C3PO数据库连接池
1 <?xml version="1.0" encoding="UTF-8"?> 2 3 -<c3p0-config> 4 5 6 -& ...
- [NOIP2013][LGOJ P1967]货车运输
Problem Link 题目描述 A国有n座城市,编号从1到n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重 ...
- kafka producer 概要(看源码前,最好能掌握)
kafakproducer概要(看源码前,最好能理解) 摘要 kafak 被设计用来作为一个统一的平台来处理庞大的数据的实时工具,在设计上有诸多变态的要求 它必须具有高吞吐量才能支持大量事件流 ...
- Java蓝桥杯01——第一题集锦:堆煤球、购物单、哪天返回、第几天、分数
堆煤球(2016JavaB) 有一堆煤球,堆成三角棱锥形.具体: 第一层放1个, 第二层3个(排列成三角形), 第三层6个(排列成三角形), 第四层10个(排列成三角形), .... 如果一共有100 ...