题目链接:

https://www.luogu.org/problemnew/show/P1967

题目描述

A国有n座城市,编号从 1到n,城市之间有 m 条双向道路。每一条道路对车辆都有重量限制,简称限重。现在有 q 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情况下,最多能运多重的货物。

输入输出格式

输入格式:

第一行有两个用一个空格隔开的整数n,m,表示 AA 国有n 座城市和 m条道路。

接下来 mm行每行33个整数 x, y, z,每两个整数之间用一个空格隔开,表示从 x号城市到y号城市有一条限重为 zz的道路。注意: x 不等于 y,两座城市之间可能有多条道路 。

接下来一行有一个整数 q,表示有 q 辆货车需要运货。

接下来 q 行,每行两个整数 x、y,之间用一个空格隔开,表示一辆货车需要从 x 城市运输货物到 y 城市,注意:x 不等于 y 。

输出格式:

共有 q 行,每行一个整数,表示对于每一辆货车,它的最大载重是多少。如果货车不能到达目的地,输出−1。

输入输出样例

输入样例#1: 复制

4 3
1 2 4
2 3 3
3 1 1
3
1 3
1 4
1 3
输出样例#1: 复制

3
-1
3

说明

对于 30%的数据,0 < n < 1,000,0 < m < 10,000,0 < q< 1,0000<n<1,000,0<m<10,000,0<q<1,000;

对于 60%的数据,0 < n < 1,000,0 < m < 50,000,0 < q< 1,0000<n<1,000,0<m<50,000,0<q<1,000;

对于 100%的数据,0 < n < 10,000,0 < m < 50,000,0 < q< 30,000,0 ≤ z ≤ 100,0000<n<10,000,0<m<50,000,0<q<30,000,0≤z≤100,000。

思路:

首先对图建立最大生成树,

倍增LCA维护树上两点之间最小距离。

注意有重边和多个连通分量的情况。

#include<bits/stdc++.h>
using namespace std;
#define mp make_pair
const int N=10010,M=50010;
struct edge{
int x,y,c;
}ed[M];
bool cmp(edge &a,edge &b){
return a.c>b.c;
}
int f[N];
int findroot(int x){
if(f[x]==x) return x;
return f[x]=findroot(f[x]);
}
void merge(int a,int b){
int f1=findroot(a),f2=findroot(b);
if(f1!=f2){
f[f1]=f2;
}
}
int n,m;
vector<pair<int,int>> a[N];
map<pair<int,int>,int > ump;
int d[N],fa[N][31],v[N][31],lg[N];
void dfs(int u,int last){
d[u]=d[last]+1;
v[u][0]=ump[mp(u,last)];
fa[u][0]=last;
for(int i=1;i<=lg[d[u]]-1;i++){
fa[u][i]=fa[fa[u][i-1]][i-1];
v[u][i]=min(v[u][i-1],v[fa[u][i-1]][i-1]);
} for(auto to:a[u]){
if(to.first!=last){
dfs(to.first,u);
}
}
} int lca(int x,int y){
int dis=0x3f3f3f3f;
if(d[x]<d[y]) swap(x,y);
while(d[x]>d[y]){
dis=min(dis,v[x][lg[d[x]-d[y]]-1]);
x=fa[x][lg[d[x]-d[y]]-1];
}
if(x==y) return dis;
for(int i=lg[d[x]]-1;i>=0;i--){
if(fa[x][i]!=fa[y][i]){
dis=min(dis,min(v[x][i],v[y][i]));
x=fa[x][i];
y=fa[y][i];
}
} return min(dis,min(v[x][0],v[y][0]));
}
int main(){
scanf("%d %d",&n,&m);
for(int i=0;i<m;i++){
int x,y,c;
if(x==y) continue;
scanf("%d %d %d",&ed[i].x,&ed[i].y,&ed[i].c);
}
sort(ed,ed+m,cmp);
for(int i=1;i<=n;i++){
f[i]=i;
lg[i]=lg[i-1]+(1<<lg[i-1]==i);//log2(i)+1;
} int cnt=0;
for(int i=0;i<m;i++){
int x=ed[i].x,y=ed[i].y;
int f1=findroot(x),f2=findroot(y);
if(f1!=f2){
f[f1]=f2;
a[x].push_back(mp(y,ed[i].c));
a[y].push_back(mp(x,ed[i].c));
ump[mp(x,y)]=ump[mp(y,x)]=ed[i].c;
cnt++;
if(cnt==n-1) break;
}
} //多个连通分量
for(int i=1;i<=n;i++){
if(f[i]==i){
ump[mp(i,0)]=0;
dfs(i,0);
}
} int Q;
scanf("%d",&Q);
for(int i=1;i<=Q;i++){
int x,y;
scanf("%d %d",&x,&y);
if(findroot(x)!=findroot(y)) printf("-1\n");
else printf("%d\n",lca(x,y));
} return 0;
}

P1967 货车运输(倍增LCA,生成树)的更多相关文章

  1. 洛谷P3379lca,HDU2586,洛谷P1967货车运输,倍增lca,树上倍增

    倍增lca板子洛谷P3379 #include<cstdio> struct E { int to,next; }e[]; ],anc[][],log2n,deep[],n,m,s,ne; ...

  2. 洛谷P1967货车运输——倍增LCA

    题目:https://www.luogu.org/problemnew/show/P1967 就是倍增LCA的裸题,注意一些细节即可. 代码如下: #include<iostream> # ...

  3. Luogu P1967 货车运输 倍增+最大生成树

    看见某大佬在做,决定补一发题解$qwq$ 首先跑出最大生成树(注意有可能不连通),然后我们要求的就是树上两点间路径上的最小边权. 我们用倍增的思路跑出来$w[u][j]$,表示$u$与的它$2^j$的 ...

  4. [洛谷 P1967] 货车运输 (最大生成树 lca)

    题目描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情况下,最多 ...

  5. NOIP2013 D1T3 货车运输 倍增LCA OR 并查集按秩合并

    思路: Kruskal求最大生成树+倍增LCA // by SiriusRen #include <cstdio> #include <cstring> #include &l ...

  6. P1967 货车运输【LCA】【生成树】

    题目描述 A 国有 nn 座城市,编号从 11 到 nn,城市之间有 mm 条双向道路.每一条道路对车辆都有重量限制,简称限重. 现在有 qq 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的 ...

  7. 「NOIP2013」「LuoguP1967」货车运输(最大生成树 倍增 LCA

    题目描述 AA国有nn座城市,编号从 11到nn,城市之间有 mm 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 qq 辆货车在运输货物, 司机们想知道每辆车在不超过车辆限重的情况下,最 ...

  8. NOIP2013 货车运输(最大生成树,倍增)

    NOIP2013 货车运输(最大生成树,倍增) A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物,司机们想知道 ...

  9. Luogu P1967 货车运输(Kruskal重构树)

    P1967 货车运输 题面 题目描述 \(A\) 国有 \(n\) 座城市,编号从 \(1\) 到 \(n\) ,城市之间有 \(m\) 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 \ ...

  10. 洛谷 P1967 货车运输

    洛谷 P1967 货车运输 题目描述 A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路.每一条道路对车辆都有重量限制,简称限重.现在有 q 辆货车在运输货物, 司机们想知道每辆车在 ...

随机推荐

  1. 移动端 canvas统计图

    一.折线图 1. 获取画布画笔 2. 封装画线的方法 3. 画坐标轴 4. 循环数据画横轴 > 4.1 画刻度 > 4.2 刻度文字 > 4.3 画折线 > 4.4 画点 5. ...

  2. 个人微信公众号搭建Python实现 -个人公众号搭建-运行run方法的编写(14.3.3)

    @ 目录 1.主要逻辑 2.代码 关于作者 1.主要逻辑 使用的是flask服务器 就使用一个函数处理请求 第一个是验证服务器,返回微信服务器给的字符串就表示验证成功 第二是要处理微信服务器发送过来的 ...

  3. zabbix学习(一)——LNMP环境搭建及zabbix安装

    第一部分:LNMP环境搭建 一.环境说明: OS:   centos7.6_x64nginx:nginx-1.16.0php:   php-7.1.11mysql:mysql-5.6.44 zabbi ...

  4. 类似818tu.c微信小说分销系统设计之多公众号网页授权自动登录源码分享

    /** 转载请保留原地址以及版权声明,请勿恶意修改 *  作者:杨浩瑞  QQ:1420213383  独立博客:http://www.yxxrui.cn * [后台]http://xiaoshuo. ...

  5. 个人MySQL的事务特性原理学习笔记总结

    目录 个人MySQL的事务特性原理笔记总结 一.基础概念 2. 事务控制语句 3. 事务特性 二.原子性 1. 原子性定义 2. 实现 三.持久性 1. 定义 2. 实现 3. redo log存在的 ...

  6. Java安全之初探weblogic T3协议漏洞

    Java安全之初探weblogic T3协议漏洞 文章首发自安全客:Java安全之初探weblogic T3协议漏洞 0x00 前言 在反序列化漏洞里面就经典的还是莫过于weblogic的反序列化漏洞 ...

  7. rtmp向IR601移植过程(无功能步骤,只有移植步骤)

    1.main.c中添加头文件: #include "rtmp_sys.h" #include "log.h" #include "rtmp.h&quo ...

  8. [leetcode]508. Most Frequent Subtree Sum二叉树中出现最多的值

    遍历二叉树,用map记录sum出现的次数,每一个新的节点都统计一次. 遍历完就统计map中出现最多的sum Map<Integer,Integer> map = new HashMap&l ...

  9. MySQL索引优化,explain详细讲解

    前言:这篇文章主要讲 explain 如何使用,还有 explain 各种参数概念,之后会讲优化 一.Explain 用法 模拟Mysql优化器是如何执行SQL查询语句的,从而知道Mysql是如何处理 ...

  10. 使用jmeter进行压力测试与nginx连接数优化

    案例训练目标 学会使用jmeter工具 学会配置nginx连接数优化 包含技能点 使用jmeter做压力测试 配置nginx的并发连接数 环境要求 PC支持VT,4G内存以上:vmware虚拟机安装有 ...