洛谷p1967货车运输(kruskal重构树)
题解中有很多说最优解是kruskal重构树
所以
抽了个早自习看了看这方面的内容
感觉真的挺好使的
首先对于kruskal算法来说
是基于贪心的思想把边权排序用并查集维护是否是在同一棵树上
对于kruskal重构树来说
按不同边权顺序排序可相应的得到最大边权的最小值 、最小边权的最大值等问题
建树过程:
排好序后, 遍历, 若两条边u, v不在同一并查集内, 那么就新建一个节点, 这个节点的点权就代表u到v的边权, 同时将这三个点都加入同一并查集
需要注意
最后建立出来的可能是森林,也就是并不是所有的点的根都是相同的
此时
可以用一个vis数组来确保所有的点都被遍历过了
1.若按照边权的升序排列
那么最后可以求得最大边权的最小值
感性理解一下
因为最后是要求最小值
所以
建一棵最小生成树
此时最大的边权就是所有别的建树方法中的最小值
2.若按照边权的降序排列
同上
要求最小边权的最大值
最后答案是最大值
那么建一棵最大生成树
关于为什么lca是答案:
感性理解一下
如果我们建的树是最小生成树
那么
显然边权越大深度越深
我们是想按着边权尽量小的走
lca(u, v)是原图u->v节点深度最小的点
所以就是他了
关于此题的代码:
#include <cstdio>
#include <iostream>
#include <algorithm>
using namespace std;
const int N = ;
int n, m, sum, fa[N], cnt, size[N], top[N], dad[N], head[N], va[N], tot, q, deep[N], vis[N], son[N];
struct Edge{
int from, to, w;
}e[N << ];
struct Node {
int nxt, to, w;
}ee[N];
void add(int x, int y) {
ee[++tot].nxt = head[x];
ee[tot].to = y;
head[x] = tot;
}
bool cmp(Edge x, Edge y) {
return x.w > y.w;
}
int find(int x) {
return fa[x] == x ? x : fa[x] = find(fa[x]);
}
void dfs(int u, int pa) {
size[u] = ;vis[u] = ;
for(int i = head[u]; i; i = ee[i].nxt) {
int v = ee[i].to;
if(v == pa) continue;
deep[v] = deep[u] + , dad[v] = u;
dfs(v, u);
size[u] += size[v];
if(size[v] > size[son[u]]) son[u] = v;
}
}
void dfs1(int u, int tp) {
top[u] = tp;
if(son[u]) dfs1(son[u], tp);
for(int i = head[u]; i; i = ee[i].nxt) {
int v = ee[i].to;
if(v == dad[u] || v == son[u]) continue;
dfs1(v, v);
}
}
int lca(int x, int y) {
while(top[x] != top[y]) {
if(deep[top[x]] < deep[top[y]]) swap(x, y);
x = dad[top[x]];
}
return deep[x] > deep[y] ? y : x;
}
int main() {
scanf("%d%d", &n, &m);
for(int i = ; i <= n; i++) fa[i] = i;
cnt = n;
for(int i = ; i <= m; i++)
scanf("%d%d%d", &e[i].from, &e[i].to, &e[i].w);
sort(e + , e + + m, cmp);
for(int i = ; i <= m; i++) {
int fx = find(e[i].from), fy = find(e[i].to);
if(fx != fy) {
va[++cnt] = e[i].w;
fa[fx] = fa[fy] = fa[cnt] = cnt;
add(fx, cnt);add(cnt, fx);
add(fy, cnt);add(cnt, fy);
}
}
for(int i = ; i <= cnt; i++)
if(!vis[i]) {
int f = find(i);
dfs(f, ), dfs1(f, f);
}
/*for(int i = 1; i <= cnt; i++)
printf("%d ", size[i]);*/
scanf("%d", &q);
while(q--) {
scanf("%d%d", &n, &m);
if(find(n) != find(m)) printf("-1\n");
else printf("%d\n", va[lca(n, m)]);
}
return ;
}
谢谢收看, 祝身体健康!
洛谷p1967货车运输(kruskal重构树)的更多相关文章
- Luogu P1967 货车运输(Kruskal重构树)
P1967 货车运输 题面 题目描述 \(A\) 国有 \(n\) 座城市,编号从 \(1\) 到 \(n\) ,城市之间有 \(m\) 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 \ ...
- 洛谷 P1967 货车运输
洛谷 P1967 货车运输 题目描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在 ...
- 洛谷P3379lca,HDU2586,洛谷P1967货车运输,倍增lca,树上倍增
倍增lca板子洛谷P3379 #include<cstdio> struct E { int to,next; }e[]; ],anc[][],log2n,deep[],n,m,s,ne; ...
- NOIP 2013 提高组 洛谷P1967 货车运输 (Kruskal重构树)
题目: A 国有 nn 座城市,编号从 11 到 nn,城市之间有 mm 条双向道路.每一条道路对车辆都有重量限制,简称限重. 现在有 qq 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情 ...
- 【杂题总汇】NOIP2013(洛谷P1967) 货车运输
[洛谷P1967] 货车运输 重做NOIP提高组ing... +传送门-洛谷P1967+ ◇ 题目(copy from 洛谷) 题目描述 A国有n座城市,编号从1到n,城市之间有m条双向道路.每一条道 ...
- [洛谷P4768] [NOI2018]归程 (kruskal重构树模板讲解)
洛谷题目链接:[NOI2018]归程 因为题面复制过来有点炸格式,所以要看题目就点一下链接吧\(qwq\) 题意: 在一张无向图上,每一条边都有一个长度和海拔高度,小\(Y\)的家在\(1\)节点,并 ...
- 洛谷P4768 [NOI2018]归程(Kruskal重构树)
题意 直接看题目吧,不好描述 Sol 考虑暴力做法 首先预处理出从$1$到每个节点的最短路, 对于每次询问,暴力的从这个点BFS,从能走到的点里面取$min$ 考虑如何优化,这里要用到Kruskal重 ...
- 洛谷P4197 Peaks (Kruskal重构树)
读题,只经过困难值小于等于x的路径,容易想到用Kruskal重构树:又要查询第k高的山峰,我们选择用主席树求解. 先做一棵重构树,跑一遍dfs,重构树中每一个非叶子节点对应一段区间,我们开range[ ...
- NOI2018 D1T1 洛谷P4768 归程 (Kruskal重构树)
实际上是一个最短路问题,但加上了海拔这个条件限制,要在海拔<水位线p中找最短路. 这里使用Kruskal重构树,将其按海拔建成小根堆,我们就可以在树中用倍增找出他不得不下车的点:树中节点有两个权 ...
随机推荐
- 示例:WPF仿制OSK做的系统键盘和数字键盘
原文:示例:WPF仿制OSK做的系统键盘和数字键盘 一.目的:在应用osk.exe系统键盘时遇到很多不方便,比如有些系统调用不出来等问题,由此开发了一个系统键盘仿制osk 二.实现功能 1.目前实现大 ...
- SAP T CODE : Description (Program)
SAP T CODE : Description (Program) V : Quickstart RKCOWUSL (RKCOWUSL)V+01 : Create Sales Call (SAPMV ...
- mysql数据库事务
事务: 一个或者一组sql语句组成一个执行的单元,这个单元要么全都执行,要么都不执行.也就是每个sql语句相互依赖.如果中间有一条出现错误则整个单元将回滚.(回滚就是刚刚的操作都撤销) 事务的属性: ...
- windows中Crontab的使用
一.jdk的安装 安装地址ttps://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 二 . ...
- 《高性能javascript》随笔
1.css文件在head标签中引入,保证在渲染结构的时候进行样式渲染2.Js文件放在body的底部,确保在渲染dom树的时候不会出现js阻塞3.函数内的变量是访问速度最快的,全局变量的访问速度是最慢的 ...
- Python - 实现矩阵转置
有个朋友提出了一个问题:手头上现在有一个二维列表,比如[[1, 2, 3], [4, 5, 6], [7, 8, 9], [10, 11, 12]],现在要把该二维列表变成为[[1, 4, 7, 10 ...
- APS系统如何让企业实现“多赢”?看高博通信是怎么做的
高博通信(上海)有限公司凭籍在超精密产业中的技术积累, 强大的资金优势以及与一流大学的联合,使得其正成为国内超精密电子制造行业的领导者. 雄厚的技术实力和专业的团队赢得了波音,空客公司等国际航空器制造 ...
- Cheat Engine 作弊表框架代码
打开游戏 打开自动汇编 扫描的所有过程,这里就省略了 引用作弊表框架代码 查找使阳光减少的地址 拷贝这个地址 添加到自动汇编脚本里,并添加汇编指令 分配到当前作弊表 生成自动汇编脚本 进行激活测试 可 ...
- DRF(django-rest_framework)框架
drf执行流程,APIView,Request -继承APIView(继承自view),重写了dispatch方法 -dispatch方法:1 request对象,被重新封装了,成了新的request ...
- MySQL Execution Plan--将范围扫描转换为等值查询
将大于或小于的范围查询装换为等值查询 在生产环境,经常会遇到需要对Worker表进行多次尝试的业务,超过一定重试次数后抛弃或使用其他方式处理,在查找满足重试条件数据时,通常会使用“小于”运算符并伴随排 ...