Codevs 3287 货车运输 2013年NOIP全国联赛提高组(带权LCA+并查集+最大生成树)

本文主要是介绍Codevs 3287 货车运输 2013年NOIP全国联赛提高组(带权LCA+并查集+最大生成树),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

3287 货车运输 2013年NOIP全国联赛提高组
时间限制: 1 s
空间限制: 128000 KB
题目等级 : 钻石 Diamond
传送门
题目描述 Description
A 国有 n 座城市,编号从 1 到 n,城市之间有 m 条双向道路。每一条道路对车辆都有重量限制,简称限重。现在有 q 辆货车在运输货物,司机们想知道每辆车在不超过车辆限重的情况下,最多能运多重的货物。
输入描述 Input Description
第一行有两个用一个空格隔开的整数 n,m,表示 A 国有 n 座城市和 m 条道路。
接下来 m 行每行 3 个整数 x、y、z,每两个整数之间用一个空格隔开,表示从 x 号城市到 y 号城市有一条限重为 z 的道路。注意:x 不等于 y,两座城市之间可能有多条道路。
接下来一行有一个整数 q,表示有 q 辆货车需要运货。
接下来 q 行,每行两个整数 x、y,之间用一个空格隔开,表示一辆货车需要从 x 城市运输货物到 y 城市,注意:x 不等于 y。
输出描述 Output Description
输出共有 q 行,每行一个整数,表示对于每一辆货车,它的最大载重是多少。如果货车不能到达目的地,输出-1。
样例输入 Sample Input
4 3
1 2 4
2 3 3
3 1 1
3
1 3
1 4
1 3
样例输出 Sample Output
3
-1
3
数据范围及提示 Data Size & Hint
对于 30%的数据,0 < n < 1,000,0 < m < 10,000,0 < q < 1,000;
对于 60%的数据,0 < 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,000。
分类标签 Tags
最小生成树 图论 大陆地区 NOIP全国联赛提高组 2013年

/*
spfa暴力.
30.
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#define MAXN 10001
#define MAXM 100001
using namespace std;
int head[MAXN],dis[MAXN],n,m,cut,k;
struct data{int v,next,x;}e[MAXM];
bool b[MAXN];
queue<int>q;
void add(int u,int v,int z)
{e[++cut].v=v;e[cut].x=z;e[cut].next=head[u];head[u]=cut;
}
int read()
{int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')f=-1;if(ch==EOF) return -1;ch=getchar();}while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar();return x*f;
}
void spfa(int s)
{memset(dis,0,sizeof dis);dis[s]=1e9;q.push(s);while(!q.empty()){int u=q.front();q.pop();b[u]=false;for(int i=head[u];i;i=e[i].next){int v=e[i].v;if(dis[v]<min(dis[u],e[i].x)) {dis[v]=min(dis[u],e[i].x);if(!b[v]) b[v]=true,q.push(v);}}}
}
int main()
{int x,y,z;n=read(),m=read();while(m--){x=read(),y=read(),z=read();add(x,y,z),add(y,x,z);}k=read();while(k--){x=read(),y=read();spfa(x);if(!dis[y]) printf("-1\n");else printf("%d\n",dis[y]);}return 0;
}
/*
最大生成树+带权LCA+并查集. 
好题.
每次询问任意两点之间边权最大路径的最小值.
▲读入的时候不能建边.
先kruskal建一棵树并保存数的形态.(prim只能求权值无法保存形态). 
搞的时候维护一个dis[i][j]数组表示从i号节点向上2^j层的到i的最小值.
然后LCA的时候步步取小,步步取小,步步取小.
若deep[i]==0&&i!=1 即改点没有遍历到,return -1. 
*/
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#define MAXN 50001
#define D 16
using namespace std;
int head[MAXN],n,m,cut,tot,father[MAXN],fa[MAXN][D+5],deep[MAXN],dis[MAXN][D+5];
struct edge
{int v;int next;int x;
}
e[MAXN<<1];
struct data
{int x,y,z;
}
s[MAXN];
void add(int u,int v,int z)
{tot++;e[tot].x=z;e[tot].v=v;e[tot].next=head[u];head[u]=tot;
}
int ffind(int x)
{if(x!=father[x])  return father[x]=ffind(father[x]);return x;
}
bool cmp(const data &x,const data &y)
{return x.z>y.z;
}
void dfs(int u,int x)
{for(int i=head[u];i!=-1;i=e[i].next){int v=e[i].v;if(!deep[v]&&v!=1){deep[v]=deep[u]+1;fa[v][0]=u;dis[v][0]=e[i].x;//dis处理.dfs(v,x+1);}}
}
void get_father()
{for(int j=1;j<=D;j++)for(int i=1;i<=n;i++){fa[i][j]=fa[fa[i][j-1]][j-1];dis[i][j]=min(dis[i][j-1],dis[fa[i][j-1]][j-1]);}
}
int get_same(int u,int v)
{for(int i=0;i<=D;i++)if((1<<i)&v) {tot=min(tot,dis[u][i]);u=fa[u][i];}return u;
}
int lca(int u,int v)
{if((u!=1&&!deep[u])||(v!=1&&!deep[v]))  return -1;tot=1e9;if(deep[u]<deep[v]) swap(u,v);u=get_same(u,deep[u]-deep[v]);if(u==v) return tot;for(int i=D;i>=0;i--){if(fa[u][i]!=fa[v][i]){tot=min(tot,dis[u][i]);tot=min(tot,dis[v][i]);u=fa[u][i];v=fa[v][i];}}tot=min(tot,min(dis[u][0],dis[v][0]));return tot;
}
int main()
{memset(head,-1,sizeof(head));memset(dis,127/3,sizeof(dis));int x,y,z,u,v;scanf("%d%d",&n,&m);for(int i=1;i<=n;i++)  father[i]=i;for(int i=1;i<=m;i++){scanf("%d%d%d",&x,&y,&z);//add(x,y,z);s[i].x=x;s[i].y=y;s[i].z=z;}sort(s+1,s+m+1,cmp);for(int i=1;i<=m;i++){u=s[i].x,v=s[i].y;int r1=ffind(u),r2=ffind(v);if(r1!=r2){add(u,v,s[i].z);add(v,u,s[i].z);father[r2]=r1;cut++;if(cut==n-1) break;}}dfs(1,0);get_father();scanf("%d",&m);for(int i=1;i<=m;i++){scanf("%d%d",&u,&v);printf("%d\n",lca(u,v));}return 0;
}

这篇关于Codevs 3287 货车运输 2013年NOIP全国联赛提高组(带权LCA+并查集+最大生成树)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/536176

相关文章

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

Python使用qrcode库实现生成二维码的操作指南

《Python使用qrcode库实现生成二维码的操作指南》二维码是一种广泛使用的二维条码,因其高效的数据存储能力和易于扫描的特点,广泛应用于支付、身份验证、营销推广等领域,Pythonqrcode库是... 目录一、安装 python qrcode 库二、基本使用方法1. 生成简单二维码2. 生成带 Log

Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南

《Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南》在日常数据处理工作中,我们经常需要将不同Excel文档中的数据整合到一个新的DataFrame中,以便进行进一步... 目录一、准备工作二、读取Excel文件三、数据叠加四、处理重复数据(可选)五、保存新DataFram

SpringBoot生成和操作PDF的代码详解

《SpringBoot生成和操作PDF的代码详解》本文主要介绍了在SpringBoot项目下,通过代码和操作步骤,详细的介绍了如何操作PDF,希望可以帮助到准备通过JAVA操作PDF的你,项目框架用的... 目录本文简介PDF文件简介代码实现PDF操作基于PDF模板生成,并下载完全基于代码生成,并保存合并P

如何提高Redis服务器的最大打开文件数限制

《如何提高Redis服务器的最大打开文件数限制》文章讨论了如何提高Redis服务器的最大打开文件数限制,以支持高并发服务,本文给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录如何提高Redis服务器的最大打开文件数限制问题诊断解决步骤1. 修改系统级别的限制2. 为Redis进程特别设置限制

详解Java中如何使用JFreeChart生成甘特图

《详解Java中如何使用JFreeChart生成甘特图》甘特图是一种流行的项目管理工具,用于显示项目的进度和任务分配,在Java开发中,JFreeChart是一个强大的开源图表库,能够生成各种类型的图... 目录引言一、JFreeChart简介二、准备工作三、创建甘特图1. 定义数据集2. 创建甘特图3.

AI一键生成 PPT

AI一键生成 PPT 操作步骤 作为一名打工人,是不是经常需要制作各种PPT来分享我的生活和想法。但是,你们知道,有时候灵感来了,时间却不够用了!😩直到我发现了Kimi AI——一个能够自动生成PPT的神奇助手!🌟 什么是Kimi? 一款月之暗面科技有限公司开发的AI办公工具,帮助用户快速生成高质量的演示文稿。 无论你是职场人士、学生还是教师,Kimi都能够为你的办公文

poj1330(LCA最近公共祖先)

题意:求最近公共祖先 思路:之前学习了树链剖分,然后我就用树链剖分的一小部分知识就可以解这个题目了,记录每个结点的fa和depth。然后查找时,每次将depth大的结点往上走直到x = y。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring>

pdfmake生成pdf的使用

实际项目中有时会有根据填写的表单数据或者其他格式的数据,将数据自动填充到pdf文件中根据固定模板生成pdf文件的需求 文章目录 利用pdfmake生成pdf文件1.下载安装pdfmake第三方包2.封装生成pdf文件的共用配置3.生成pdf文件的文件模板内容4.调用方法生成pdf 利用pdfmake生成pdf文件 1.下载安装pdfmake第三方包 npm i pdfma

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n