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 ...
随机推荐
- CodeForces 1311E Construct the Binary Tree
题意 给定\(n\)和\(d\),构造一颗\(n\)个节点的二叉树(以\(1\)为根),所有节点到\(1\)的距离和为\(d\),不行输出\(NO\),否则输出\(YES\)和\(2\)-\(n\)的 ...
- 蚂蚁集团混沌工程 ChaosMeta V0.5 版本发布
混沌工程 ChaosMeta 的全新版本 V0.5 现已正式发布!该版本包含了许多新特性和增强功能,为用户提供了支撑混沌工程各个阶段的平台能力,以及降低使用门槛的用户界面. ChaosMeta V0. ...
- iOS开发过程中的内存监控
- [项目源码] JavaWeb校园宿舍管理系统
jsp校园宿舍管理系统源码,采用Servlet+JSP+MySQL.包含数据库文件,界面采用bootstrap,简洁大方. 项目导入eclipse后的目录结构如下: 关注下面公众号,下载源码
- SQL注入简介
SQL注入(SQL Injection)是一种计算机安全漏洞,它允许攻击者通过操纵应用程序的输入来执行恶意的SQL查询,从而访问.修改或删除数据库中的数据.这种攻击通常发生在应用程序未正确验证.过滤或 ...
- c语言代码练习7
//输出0-100的三倍数字#define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> int main() { int i = 0; int ...
- Java-全网最详细数据结构
数构&算法:数据结构 数据结构是计算机存储.组织数据的方式.数据结构是指相互之间存在一种或多种特定关系的数据元素的集合.通常情况下,精心选择的数据结构可以带来更高的运行或者存储效率.数据结构往 ...
- 【虹科干货】Redis 开发者需要了解的缓存驱逐策略
在你搭建并配置了一个Redis数据库之后,Redis成功地提升了应用程序性能.然而这里有一个潜在问题,随着缓存数据的快速增加和内存占用率的逐渐上升,你很快会发现Redis缓存容量即将达到硬件存储容量上 ...
- SpringBoot数据响应、分层解耦、三层架构
响应数据 @ResponseBody 类型:方法注解.类注解 位置:Controller方法.类上 作用:将方法返回值直接响应,如果返回值类型是 实体对象/集合 ,将会转换为json格式响应 说明:@ ...
- 该如何选择ClickHouse的表引擎
该如何选择ClickHouse的表引擎 本文将介绍ClickHouse中一个非常重要的概念-表引擎(table engine).如果对MySQL熟悉的话,或许你应该听说过InnoDB和MyISAM存储 ...