【BZOJ】3732: Network【Kruskal重构树】
3732: Network
Time Limit: 10 Sec Memory Limit: 128 MB
Submit: 2812 Solved: 1363
[Submit][Status][Discuss]
Description
给你N个点的无向图 (1 <= N <= 15,000),记为:1…N。
图中有M条边 (1 <= M <= 30,000) ,第j条边的长度为: d_j ( 1 < = d_j < = 1,000,000,000).
现在有 K个询问 (1 < = K < = 20,000)。
每个询问的格式是:A B,表示询问从A点走到B点的所有路径中,最长的边最小值是多少?
Input
第一行: N, M, K。
第2..M+1行: 三个正整数:X, Y, and D (1 <= X <=N; 1 <= Y <= N). 表示X与Y之间有一条长度为D的边。
第M+2..M+K+1行: 每行两个整数A B,表示询问从A点走到B点的所有路径中,最长的边最小值是多少?
Output
对每个询问,输出最长的边最小值是多少。
Sample Input
1 2 5
2 3 4
3 4 3
1 4 8
2 5 7
4 6 2
1 2
1 3
1 4
2 3
2 4
5 1
6 2
6 1
Sample Output
5
5
4
4
7
4
5
HINT
1 <= N <= 15,000
1 <= M <= 30,000
1 <= d_j <= 1,000,000,000
1 <= K <= 15,000
Solution
看到题想到的是前几年noip考的货车运输,就是建出最大生成树再链剖+线段树求链上最小边即可,学习了一波$Kruskal$重构树后,整道题就直接变成最大生成树+求LCA了!
这道题是模板了,大佬%%%写的相当清楚了,在最小生成树连接两个块时,新建一个点,作为两个块最远祖先(并查集中)的父亲节点,把这条边的权值下放到新建节点的点权,因为这条边就是连接两个块的最长边权的最小值,所以每次询问返回两个点的LCA的点权值即可。
再也不用复杂的线段树或者倍增啦!!!
Code
- #include<bits/stdc++.h>
- using namespace std;
- int n, m, k;
- struct Node {
- int u, v, nex, w;
- } Edge1[], Edge[];
- bool cmp(Node a, Node b) { return a.w < b.w; }
- int stot1;
- void add1(int u, int v, int w) {
- Edge1[++stot1] = (Node) {u, v, , w};
- }
- int stot, h[];
- void add(int u, int v) {
- Edge[++stot] = (Node) {u, v, h[u], };
- h[u] = stot;
- }
- int u_fa[];
- int find(int u) {
- if(u != u_fa[u]) u_fa[u] = find(u_fa[u]);
- return u_fa[u];
- }
- int t, val[];
- void Kruskal() {
- for(int i = ; i <= n * ; i ++) u_fa[i] = i;
- t = n;
- for(int i = ; i <= m; i ++) {
- int u = Edge1[i].u, v = Edge1[i].v, w = Edge1[i].w;
- int uu = find(u), vv = find(v);
- if(uu != vv) {
- u_fa[uu] = ++ t; u_fa[vv] = t;
- add(t, uu); add(t, vv); val[t] = w;
- }
- }
- }
- int siz[], fa[], son[], dep[];
- void dfs1(int u, int ff) {
- fa[u] = ff; siz[u] = ; dep[u] = dep[ff] + ;
- for(int i = h[u]; i; i = Edge[i].nex) {
- int v = Edge[i].v;
- if(v == ff) continue;
- dfs1(v, u);
- siz[u] += siz[v];
- if(siz[v] > siz[son[u]]) son[u] = v;
- }
- }
- int top[];
- void dfs2(int u, int tp) {
- top[u] = tp;
- if(son[u]) dfs2(son[u], tp);
- for(int i = h[u]; i; i = Edge[i].nex) {
- int v = Edge[i].v;
- if(v == son[u] || v == fa[u]) continue;
- dfs2(v, v);
- }
- }
- int LCA(int u, int v) {
- if(find(u) != find(v)) return ;
- while(top[u] != top[v]) {
- if(dep[top[u]] < dep[top[v]]) swap(u, v);
- u = fa[top[u]];
- }
- if(dep[u] < dep[v]) swap(u, v);
- return val[v];
- }
- int main() {
- scanf("%d%d%d", &n, &m, &k);
- for(int i = ; i <= m; i ++) {
- int u, v, w;
- scanf("%d%d%d", &u, &v, &w);
- add1(u, v, w);
- }
- sort(Edge1 + , Edge1 + + m, cmp);
- Kruskal();
- for(int i = ; i <= t; i ++)
- if(!dep[i]) {
- dfs1(u_fa[i], ); dfs2(u_fa[i], u_fa[i]);
- }
- for(int i = ; i <= k; i ++) {
- int u, v;
- scanf("%d%d", &u, &v);
- printf("%d\n", LCA(u, v));
- }
- return ;
- }
【BZOJ】3732: Network【Kruskal重构树】的更多相关文章
- [bzoj 3732] Network (Kruskal重构树)
kruskal重构树 Description 给你N个点的无向图 (1 <= N <= 15,000),记为:1-N. 图中有M条边 (1 <= M <= 30,000) ,第 ...
- BZOJ 3732: Network Kruskal 重构树
模板题,练练手~ Code: #include <cstdio> #include <algorithm> #define N 80000 #define setIO(s) f ...
- 【BZOJ 3732】 Network Kruskal重构树+倍增LCA
Kruskal重构树裸题, Sunshine互测的A题就是Kruskal重构树,我通过互测了解到了这个神奇的东西... 理解起来应该没什么难度吧,但是我的Peaks连WA,,, 省选估计要滚粗了TwT ...
- BZOJ3732: Network(Kruskal重构树)
题意 Link 给出一张$n$个点的无向图,每次询问两点之间边权最大值最小的路径 $n \leqslant 15000, m \leqslant 30000, k \leqslant 20000$ S ...
- BZOJ 3732 Network Kruskal+倍增LCA
题目大意:给定一个n个点m条边的无向连通图.k次询问两点之间全部路径中最长边的最小值 NOIP2013 货车运输.差点儿就是原题...仅仅只是最小边最大改成了最大边最小.. . 首先看到最大值最小第一 ...
- Kruskal重构树+LCA || BZOJ 3732: Network
题面:https://www.lydsy.com/JudgeOnline/problem.php?id=3732 题解:Kruskal重构树板子 代码: #include<cstdio> ...
- BZOJ 3732 Network 【模板】kruskal重构树
[题解] 首先,我们可以发现,A到B的所有路径中,最长边的最小值一定在最小生成树上.我们用Kruskal最小生成树时,假设有两个点集U,V,若加入一条边w(u,v)使U,V联通,那么w就是U中每个点到 ...
- Kruskal重构树学习笔记+BZOJ3732 Network
今天学了Kruskal重构树,似乎很有意思的样子~ 先看题面: BZOJ 题目大意:$n$ 个点 $m$ 条无向边的图,$k$ 个询问,每次询问从 $u$ 到 $v$ 的所有路径中,最长的边的最小值. ...
- BZOJ 3551: [ONTAK2010]Peaks加强版 [Kruskal重构树 dfs序 主席树]
3551: [ONTAK2010]Peaks加强版 题意:带权图,多组询问与一个点通过边权\(\le lim\)的边连通的点中点权k大值,强制在线 PoPoQQQ大爷题解传送门 说一下感受: 容易发现 ...
随机推荐
- Spark记录-Spark作业调试
在本地IDE里直接运行spark程序操作远程集群 一般运行spark作业的方式有两种: 本机调试,通过设置master为local模式运行spark作业,这种方式一般用于调试,不用连接远程集群. 集群 ...
- Linux ------清除内存中的cache
首先以Centos6.4的来说,Centos7有些区别 一.buffer/cache/swap的介绍 #cat /etc/redhat-release #查看系统版本 CentOS release ...
- ifconfig不显示网卡eth0
参考资料:http://blog.itpub.net/25851087/viewspace-1700568/ 在/etc/sysconfig/network-script/ifcfg-eth0网卡配置 ...
- IEnumerator和IEnumerable详解
IEnumerator和IEnumerable 从名字常来看,IEnumerator是枚举器的意思,IEnumerable是可枚举的意思. 了解了两个接口代表的含义后,接着看源码: IEnumerat ...
- Nginx模块之Nginx-Ts-Module学习笔记(一)抢险体验
1.通过HTTP接收MPEG-TS2.生产和管理Live HLS 3.按照官方的编译和配置,当然了我是第一次编译没有通过,在作者重新调整下,编译成功,感谢:@arut https://github.c ...
- [iOS]深拷贝/浅拷贝区别
来点鸡汤: // 所谓拷贝 就是在原有的对象的基础上产生一个新的副本对象.有两点原则: // 1. 改变原对象的属性和行为不会影响副本对象 // 2. 改变副本对象的属性和行为不会影响原对象 ...
- MYSQL查询重复记录的方法
select * from hengtu_demandpush a where (a.did,a.mid) in (select did,mid from hengtu_demandpush grou ...
- Django之模板语法
Django框架之第三篇模板语法(重要!!!) 一.什么是模板? 只要是在html里面有模板语法就不是html文件了,这样的文件就叫做模板. 二.模板语法分类 一.模板语法之变量:语法为 {{ }}: ...
- Zookeeper笔记之quota
一.节点配额概述 zookeeper中可以往节点存放数据,但是一般来说存放数据总是要有个度量的对吧,不然空间就那么大,如果某个节点将空间全占用了其它节点没得用了,所以zookeeper提供了一个对节点 ...
- 【Python】测试dpkt解析pcap
1.前言 本想借助dpkt解析mail.dns.http来辅助分析pcap包进行分析,查阅资料学习却发现并不如使用scapy那么方便. dpkt是一个python模块,可以对简单的数据包创建/解析,以 ...