hszxoj 货车运输 [lca]
题目链接: hszxoj 货车运输
题目描述与思路
- 简化题目:
求 \(x\)到 \(y\) 两点间路径的边权最小值的最大值 - 与之前的最短路最大的不同是这道题是多源最短路,那么 \(spfa\) 就废了,\(Floyd\) 定会 \(TLE\) 所以就需要用新的算法。
- 用 \(lca\) 一定是在树上的,但明显这玩意他既有环又有森林,直接用就会 \(RE\)
- 所以我们需要用到 \(kru\) 重构树
首先展示 \(kru\) 代码
void kru()
{
stable_sort(e+1,e+1+m,cmp);
cnt=n;
for(int i=1;i<=m;i++)
{
int a=find(e[i].x),b=find(e[i].y);
if(a==b) continue;
f[a]=f[b]=++cnt;
g[a].push_back(cnt),g[cnt].push_back(a),
g[b].push_back(cnt),g[cnt].push_back(b);
w[cnt]=e[i].w;
if(++tot==n-1) break;
}
root=n+tot;
}
- 首先排序一下,注意这道题是 \(最大生成树\) ,所以从大到小排序,这样求出来才是最小值的最大值,不然就是最大值的最小值了
- 有三个重要点
- \(1.\) 和正常的最小生成树不同,我们最后只需要输出两点的最近公共祖先的点权,但我们实则求的是对应的边权,就需要将边权转换为点权。不妨将相连的两点 \(x\),\(y\) 中间插入一个点,让这个点的点权为 \(x,y\) 的边权,并使这个点成为 \(x\) 和 \(y\) 的祖先
- 为什么让他成为这两个点的祖先也是显而易见的,我们最后输出最近公共祖先的权值,只有类似这个新点的权值,他才不是 \(0\),看完全部代码后会理解得更清晰

\(变化后是这样,5是新的点,5的权值为1\)

- 为什么让他成为这两个点的祖先也是显而易见的,我们最后输出最近公共祖先的权值,只有类似这个新点的权值,他才不是 \(0\),看完全部代码后会理解得更清晰
- \(2.\) 我们需要搞出来树根 \(root\) 代码里最后有一行 \(root=n+tot\) 为什么这么做呢?他重构树之后显然是有 \(森林\) 的,但他最后一个点不可能是森林(不然他无父无母的都遍历不到这个点),相当重要的,否则只有15分,后面 \(dfs\) 时有用
- \(3.\) 某个脑残问题,看到那个 \(break\) 没有,改成 \(return\) ,\(root\) 就没了 \(\large{qwq}\)
- \(1.\) 和正常的最小生成树不同,我们最后只需要输出两点的最近公共祖先的点权,但我们实则求的是对应的边权,就需要将边权转换为点权。不妨将相连的两点 \(x\),\(y\) 中间插入一个点,让这个点的点权为 \(x,y\) 的边权,并使这个点成为 \(x\) 和 \(y\) 的祖先
- 然后快乐的打完 \(lca\) 就完事了,最后输出两点最近公共祖先的权值即可,题中有一个若两点无法到达,输出 \(-1\) ,判断一下他们两个是不是在一个并查集里,如果不是自然就是不通
- 并查集如果用我这种方法别忘了 \(初始化\),初始化时要搞到 \(n+m\) ,因为我们是要建新的点的不然会炸内存
代码呈上
#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
const int N=5e5+1;
template<typename Tp> inline void read(Tp&x)
{
x=0;register bool z=1;
register char c=getchar();
for(;c<'0'||c>'9';c=getchar()) if(c=='-') z=0;
for(;'0'<=c&&c<='9';c=getchar()) x=(x<<1)+(x<<3)+(c^48);
x=(z?x:~x+1);
}
int n,m,q,cnt,root,tot;
int w[N],f[N],fa[N],dep[N],sz[N],son[N],top[N];
vector<int>g[N];
struct aa{int x,y,w;}e[N];
bool cmp(aa s1,aa s2){return s1.w>s2.w;}
int find(int x)
{
if(f[x]!=x) f[x]=find(f[x]);
return f[x];
}
void kru()
{
stable_sort(e+1,e+1+m,cmp);
cnt=n;
for(int i=1;i<=m;i++)
{
int a=find(e[i].x),b=find(e[i].y);
if(a==b) continue;
f[a]=f[b]=++cnt;
g[a].push_back(cnt),g[cnt].push_back(a),
g[b].push_back(cnt),g[cnt].push_back(b);
w[cnt]=e[i].w;
if(++tot==n-1) break;
}
root=n+tot;
}
void dfs1(int x,int t)
{
fa[x]=t,dep[x]=dep[t]+1,sz[x]=1;
for(int y:g[x])
if(y!=t)
{
dfs1(y,x);
sz[x]+=sz[y];
if(sz[son[x]]<sz[y]) son[x]=y;
}
}
void dfs2(int x,int t)
{
top[x]=t;
if(!son[x]) return ;
dfs2(son[x],t);
for(int y:g[x])
if(y!=son[x]&&y!=fa[x])
dfs2(y,y);
}
int lca(int x,int y)
{
for(;top[x]!=top[y];x=fa[top[x]])
if(dep[top[x]]<dep[top[y]]) swap(x,y);
return dep[x]<dep[y]?x:y;
}
signed main()
{
#ifndef ONLINE_JUDGE
freopen("in.txt","r",stdin);
freopen("out.txt","w",stdout);
#endif
int x,y;
read(n),read(m);
for(int i=1;i<=n+m;i++) f[i]=i;
for(int i=1;i<=m;i++)
read(e[i].x),read(e[i].y),read(e[i].w);
kru(),dfs1(root,0),dfs2(root,root);
read(q);
while(q--)
read(x),read(y),
cout<<((find(x)!=find(y))?-1:w[lca(x,y)])<<endl;
}
并查集还可以这么打(虽然大多数人都是这么打的)
int find(int x)
{
return (!f[x])?x:f[x]=find(f[x]);
}
这样就不需要初始化了
hszxoj 货车运输 [lca]的更多相关文章
- NOIP2013货车运输[lca&&kruskal]
题目描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情况下,最多 ...
- NOIP2013 货车运输 LCA倍增+最大生成树
#include <cstdio> #include <iostream> #include <ctime> #include <vector> #in ...
- 洛谷 P1967 货车运输 LCA + 最小生成树
两点之间边权最大值的最小值一定在图的最小生成树中取到. 求出最小生成树,进行倍增即可. Code: #include<cstdio> #include<algorithm> u ...
- NOIP2013 货车运输 (最大生成树+树上倍增LCA)
死磕一道题,中间发现倍增还是掌握的不熟 ,而且深刻理解:SB错误毁一生,憋了近2个小时才调对,不过还好一遍AC省了更多的事,不然我一定会疯掉的... 3287 货车运输 2013年NOIP全国联赛提高 ...
- Codevs 3287 货车运输 2013年NOIP全国联赛提高组(带权LCA+并查集+最大生成树)
3287 货车运输 2013年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 传送门 题目描述 Description A 国有 n 座 ...
- 「NOIP2013」「LuoguP1967」货车运输(最大生成树 倍增 LCA
题目描述 AA国有nn座城市,编号从 11到nn,城市之间有 mm 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 qq 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情况下,最 ...
- 洛谷P3379lca,HDU2586,洛谷P1967货车运输,倍增lca,树上倍增
倍增lca板子洛谷P3379 #include<cstdio> struct E { int to,next; }e[]; ],anc[][],log2n,deep[],n,m,s,ne; ...
- codevs3287货车运输(最小生成树+LCA)
3287 货车运输 2013年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description A 国有 ...
- NOIP 2013 货车运输【Kruskal + 树链剖分 + 线段树 】【倍增】
NOIP 2013 货车运输[树链剖分] 树链剖分 题目描述 Description A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在 ...
- C++之路进阶——codevs3287(货车运输)
3287 货车运输 2013年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description A 国有 n ...
随机推荐
- Linux下MySQL备份指定数据库命令
比如我们要备份mysql中已经存在的名为linux的数据库,要用到命令mysqldump 命令格式如下: [root@linuxsir01 root]# mysqldump -u root -p li ...
- vue + canvas 实现九宮格手势解锁器
前言 专栏分享:vue2源码专栏,vue router源码专栏,玩具项目专栏,硬核推荐 欢迎各位 ITer 关注点赞收藏 此篇文章用于记录柏成从零开发一个canvas九宮格手势解锁器的历程,最终效果如 ...
- MySQL面试题——隔离级别相关面试题
隔离级别相关面试题 MySQL事务隔离级别 未提交读--可以读到其他事务未提交的数据(最新的版本) 错误现象:脏读.不可重复读.幻读的现象 提交读(RC)--可以读到其他事务已提交的数据(最新已提交的 ...
- 【火坑】一切从TimeSpan说起
小编在编写WPF程序时,需要做一个判断:定时使用Modbus协议使用Quartz.net 定时任务读取设备中的数据,同时也使用定时任务判断是否长时间获取不到数据的情况,如果程序中超过一分钟没有获取到数 ...
- VOLTE:MIUI14无5G信号、通话质量差、短信收不到的原因之一
在最近一次miui版本更新后的一段时间里,先后出现以下一些情况: 1. 在"双卡与移动网络"设置中,无论如何折腾"5G网络"开关或者"5G网络模式选择 ...
- 20230919 .NET面经
SQL IQuerable 和 IEnumerable 的主要区别? https://stackoverflow.com/questions/252785/what-is-the-difference ...
- Solution -「洛谷 P5659」「CSP-S 2019」树上的数
Description Link. 联赛原题应该都读过吧-- Solution Part 0 大致思路 主要的思路就是逐个打破,研究特殊的数据得到普通的结论. Part 1 暴力的部分分 暴力的部分分 ...
- DevOps2023现状报告|注重文化、以用户为中心是成功的关键
Google Cloud DORA 团队的一份新研究报告强调了企业文化和关注用户作为成功软件交付支柱的重要性. 2023 DevOps 状况报告分析了过去 9 年来通过此类最大规模调查收集的全球 36 ...
- 非全自研可视化表达引擎RuleLinK可视化之路
导读 上一篇<非全自研可视化表达引擎-RuleLinK>介绍了RuleLink的V1.0版本,虽说一定程度上消除了一些配置相关的样板式代码,也肉眼可见的消除了一些研发资源的浪费:RuleL ...
- 基于LangChain的LLM应用开发3——记忆
此情可待成追忆,只是当时已惘然.我们人类会有很多或美好或痛苦的回忆,有的回忆会渐渐模糊,有的回忆午夜梦醒,会浮上心头. 然而现在的大语言模型都是没有记忆的,都是无状态的,大语言模型自身不会记住和你对话 ...