题目链接:https://www.luogu.org/problemnew/show/P2245

= 货车运输

被逼着写过mst+lca

后来成了mst+树剖

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int maxn = 100010;
int n, m, q, node[maxn], f[maxn];
struct mst{
int u, v, w;
}ee[maxn*3];
struct edge{
int next, to, len;
}e[maxn*3];
int head[maxn], cnt;
void add(int u, int v, int w)
{
e[++cnt].len = w; e[cnt].next = head[u]; e[cnt].to = v; head[u] = cnt;
e[++cnt].len = w; e[cnt].next = head[v]; e[cnt].to = u; head[v] = cnt;
}
//-----Kruskal-----
bool cmp(mst a, mst b)
{
return a.w > b.w;
}
int find(int x)
{
return f[x] == x ? x : f[x] = find(f[x]);
}
void Kruskal()
{
sort(ee+1, ee+1+m, cmp);
for(int i = 1; i <= m; i++)
{
if(find(ee[i].u) == find(ee[i].v)) continue;
add(ee[i].v, ee[i].u, ee[i].w);
f[find(ee[i].v)] = f[find(ee[i].u)];
}
}
//----------------
int top[maxn], size[maxn], son[maxn], fa[maxn], seg[maxn], rev[maxn], deep[maxn], num = -1;
void dfs(int u, int f)
{
fa[u] = f;
for(int i = head[u]; i != -1; i = e[i].next)
{
if(e[i].to == fa[u]) continue;
node[e[i].to] = e[i].len;
dfs(e[i].to, u);
}
}
void dfs1(int u)
{
int maxson = -1;
deep[u] = deep[fa[u]]+1;
size[u] = 1;
son[u] = n+1;
for(int i = head[u]; i != -1; i = e[i].next)
{
int v = e[i].to;
if(v == fa[u]) continue;
dfs1(v);
size[u] += size[v];
if(maxson < size[v]) maxson = size[v], son[u] = v;
}
}
void dfs2(int u, int t)
{
top[u] = t;
num++;
seg[u] = num;
rev[num] = u;
if(son[u] != n+1) dfs2(son[u], t);
for(int i = head[u]; i != -1; i = e[i].next)
{
if(e[i].to == fa[u] || e[i].to == son[u]) continue;
dfs2(e[i].to, e[i].to);
}
}
//-------------------
class Segment_Tree{
#define lson l, mid, rt<<1
#define rson mid+1, r, rt<<1|1
public:
int tree[maxn<<2];
void build(int l, int r, int rt)
{
if(l == r)
{
tree[rt] = node[rev[l]];
return;
}
int mid = (l + r) >> 1;
build(lson);
build(rson);
PushUP(rt);
}
int query(int left, int right, int l, int r, int rt)
{
int res = 0x7fffffff;
if(left <= l && r <= right) return tree[rt];
int mid = (l + r) >> 1;
if(left <= mid) res = min(res, query(left, right, lson));
if(right > mid) res = min(res, query(left, right, rson));
return res;
}
int Qrange(int x, int y)
{
int ret = 0x7fffffff;
while(top[x] != top[y])
{
if(deep[top[x]] < deep[top[y]]) swap(x, y);
ret = min(ret, query(seg[top[x]], seg[x], 1, n, 1));
x = fa[top[x]];
}
if(deep[x] > deep[y]) swap(x, y);
ret = min(ret, query(seg[x]+1, seg[y], 1, n, 1));
return ret;
}
private:
void PushUP(int rt)
{
tree[rt] = min(tree[rt<<1], tree[rt<<1|1]);
}
}T;
int main()
{
ios::sync_with_stdio(false);
memset(head, -1, sizeof(head));
cin>>n>>m;
for(int i = 1; i <= m; i++) cin>>ee[i].u>>ee[i].v>>ee[i].w;
for(int i = 1; i <= n; i++) f[i] = i;
Kruskal();
memset(fa, -1, sizeof(fa));
fa[0] = 0;
for(int i = 1; i <= n; i++) if(fa[i] == -1) dfs(i, 0);
for(int i = 1; i <= n; i++) if(fa[i] == 0) add(i, 0, 0);
dfs1(0);
dfs2(0, 0);
T.build(1, n, 1);
cin>>q;
for(int i = 1; i <= q; i++)
{
int x, y;
cin>>x>>y;
if(find(x) != find(y)) cout<<"-1\n";
else cout<<T.Qrange(x, y)<<"\n";
}
return 0;
}

【luogu P2245 星际导航】 题解的更多相关文章

  1. [LUOGU] P2245 星际导航

    题目描述 sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为了方便起见,我们可以认为宇宙是一张有N 个顶点和M 条边的带权无向图,顶点表示各个星系, ...

  2. Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集)

    Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集) Description sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为 ...

  3. 洛谷 P2245 星际导航 解题报告

    P2245 星际导航 题目描述 sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为了方便起见,我们可以认为宇宙是一张有N 个顶点和M 条边的带权无向 ...

  4. P2245 星际导航 瓶颈路

    \(\color{#0066ff}{ 题目描述 }\) sideman 做好了回到 \(\text{Gliese}\) 星球的硬件准备,但是 \(\text{sideman}\) 的导航系统还没有完全 ...

  5. P2245 星际导航

    题目描述 sideman 做好了回到 Gliese星球的硬件准备,但是 sideman 的导航系统还没有完全设计好.为了方便起见,我们可以认为宇宙是一张有 N 个顶点和 M 条边的带权无向图,顶点表示 ...

  6. [洛谷P2245]星际导航

    题目大意:有一张n点m边的带权无向图,和一些问题,每次询问两个点之间的路径的最大边权最小是多少. 解题思路:同NOIP2013货车运输,只是数据增大,大变成小,小变成大了而已.所以具体思路见货车运输. ...

  7. 最小生成树+LCA【洛谷 P2245】 星际导航

    [洛谷 P2245] 星际导航 题目描述 sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为了方便起见,我们可以认为宇宙是一张有N 个顶点和M 条边 ...

  8. Luogu P2754 星际转移问题

    Luogu P2754 星际转移问题 思路 首先,对于地球能否到达月球的问题,考虑使用并查集维护. 对于每艘飞船能够到达的站点,放进一个集合里,若两艘飞船的集合有交集,那么就合并两个集合,最后只要地球 ...

  9. 【洛谷P2245】星际导航

    题面 题解 \(kruskal\)重构树板子题??(大雾 因为重构树上两点之间的\(LCA\)的权值就是原图上最小生成树上的瓶颈. 所以建个重构树,跑\(LCA\)即可. 代码 #include< ...

随机推荐

  1. 搭建 Visual Studio 2012 + DXperience-13.2.6 + MySql 开发平台

    一. 开发环境 1. 此开发平台主要用来开发基于.NET 4.0及以上版本的应用 2. 点击此下载 Visual Studio 2012 Ultimate 中文版开发工具 3. 点击此下载 DXper ...

  2. C# 调用C++DLL 类型转换

    内容转自网上····这里做 备份··· 原文链接: http://blog.csdn.net/miss_easy/article/details/52470964 /C++中的DLL函数原型为 //e ...

  3. ES6学习笔记(四)-数值扩展

    PS: 前段时间转入有道云笔记,体验非常友好,所以笔记一般记录于云笔记中,每隔一段时间,会整理一下, 发在博客上与大家一起分享,交流和学习. 以下:

  4. websocket协议及案例

    WebSocket是一种用于在服务器与客户端之间实现高效的双向通信的机制.可以解决数据实时性要求比较高的应用,比如:在线聊天,在线教育,炒股或定位等. 一:websocket产生背景: 为了解决这种实 ...

  5. 一键清理 Nexus 中无用的 Docker 镜像

    现许多团队使用 Nexus 来管理 Docker 镜像,产品不断迭代,镜像仓库占用的磁盘空间也越来越大.由于 Nexus 的控制台并未提供批量操作镜像功能,清理镜像十分不便.本文分享一个清理 Nexu ...

  6. Windows下本机简易监控系统搭建(Telegraf+Influxdb+Grafana)

    一.文件准备 1.1 文件名称 telegraf-1.2.1_windows_amd64.zip influxdb-1.2.2_windows_amd64.zip grafana-4.2.0.wind ...

  7. Visual Studio 2015 + Update 1

    Visual Studio 2015是一个基本完整的开发工具集,它包括了整个软件生命周期中所需要的大部分工具,如UML工具.代码管控工具.集成开发环境(IDE)等等.所写的目标代码适用于微软支持的所有 ...

  8. LinkedList源码疑问记录

    早上看linkedList源码时候,对于它的初始化一直不太明白.如下: transient int size = 0; /** * Pointer to first node. * Invariant ...

  9. macvtap与vhost-net技术

    MacVlan的功能是给同一个物理网卡配置多个MAC地址,可以在软件上配置多个以太网口,属于物理层的功能.MacVTap是用来替代TUN/TAP和Bridge内核模块的.MacTap是基于MacVla ...

  10. c# webservice中访问http和https的wsdl,生成的配置节点的不同之处

    http: https: