图论问题(2) : hdu 1102
题目转自hdu 1102,题目传送门
题目大意:
输入一个n*n的邻接矩阵,其中i行j列代表从i到j的路径的长度
然后又m条路已经帮你修好了,求最短要修多长的路才能使所有村庄连接
不难看出,这道题就是标准的最小生成树模板,多水啊
解题思路
虽然很水,但本人还是调了近1h才把代码调好......
下面介绍一下解决最小生成树的两个方法:
Prim 和 Kruskal
一,Prim(不懂的点这里)
Prim的思想和dijkstra的想法很想(如果不知道dijkstra算法的请点这里)
那么Prim的复杂度在为优化之前是O(n2),还是很慢的(虽然这道题能过)
既然这样,那这道题该怎么用Prim解呢?
思考了近10min后我想到了一个绝妙的方法,但是这里地方太小写不下
既然已经有建好了的,那我们肯定要用他已经建好的
所以,我们在输入时做一个预处理
将所有已经建过的路的距离化为0,然后再跑一遍Prim就行了
预处理代码如下:
for(int i=;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
g[x][y]=g[y][x]=;
}
p.s.:g为邻接矩阵
然后在花15min打一遍Prim算法就可以愉快地AC了
AC代码如下:
#include<iostream>
#include<stdio.h>
#include<algorithm>
#include<string.h>
#define inf 2147483647
using namespace std;
bool vis[];
int n,m,cnt,ans,u;
int dis[];
int g[][];
void init()
{
ans=cnt=;
memset(vis,false,sizeof(vis));
memset(dis,0x7f,sizeof(dis));
return ;
}
void pirm()
{
dis[]=;
while(true)
{
u=;
for(int i=;i<=n;i++)
if(!vis[i] && (dis[i]<dis[u])) u=i;
if(u==) return ;
vis[u]=true;
ans+=dis[u];
for(int i=;i<=n;i++) dis[i]=min(dis[i],g[u][i]);
}
}
int main()
{
while(scanf("%d",&n)!=EOF)
{
init();
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
scanf("%d",&g[i][j]);
scanf("%d",&m);
for(int i=;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
g[x][y]=g[y][x]=;
}
pirm();
printf("%d\n",ans);
}
return ;
}
接下来看Kruskal......
二,Kruskal(不懂的点这里)
Kruskal中将用到hdu 1198中的并查集(点此转到我的的博客:图论问题(1):hdu 1198)
Kruskal主要就是把边按边权从小到大排序
在通过并查集检查目前最小的边的两端是否在同一集合中
若是,则跳过这条边
否则就把他们归为一个集合
这里只需要提前作这一步骤就行了
预处理代码如下:
for(int i=;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
Union(x,y);
}
p.s.:其中Union为合并函数
然后就花个20min写完模板就可以愉快地AC了
AC代码如下:
#include<iostream>
#include<stdio.h>
#include<algorithm>
using namespace std;
struct edge
{
int from,to,w;
bool operator<(const edge &a)const
{
return w<a.w;
}
}e[];
int n,m,cnt,ans;
int fa[];
void init()
{
for(int i=;i<=n;i++) fa[i]=i;
cnt=ans=;
return ;
}
int find_fa(int x)
{
if(x==fa[x]) return x;
else
{
fa[x]=find_fa(fa[x]);
return fa[x];
}
return ;
}
void Union(int x,int y)
{
x=find_fa(x);
y=find_fa(y);
if(x<y) fa[y]=x;
else fa[x]=y;
return ;
}
int main()
{
while(scanf("%d",&n)!=EOF)
{
init();
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
{
scanf("%d",&e[cnt].w);
e[cnt].from=i;e[cnt].to=j;
cnt++;
}
scanf("%d",&m);
for(int i=;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
Union(x,y);
}
sort(e,e+cnt);
for(int i=;i<cnt;i++)
if(find_fa(e[i].from)!=find_fa(e[i].to))
{
Union(e[i].from,e[i].to);
ans+=e[i].w;
}
printf("%d\n",ans);
}
return ;
}
今天的讲解就到这了,若果有没有听懂的可以借鉴一下《啊哈!算法》
图论问题(2) : hdu 1102的更多相关文章
- HDU 1102 最小生成树裸题,kruskal,prim
1.HDU 1102 Constructing Roads 最小生成树 2.总结: 题意:修路,裸题 (1)kruskal //kruskal #include<iostream> ...
- HDU 1102 Constructing Roads, Prim+优先队列
题目链接:HDU 1102 Constructing Roads Constructing Roads Problem Description There are N villages, which ...
- hdu 1102 Constructing Roads Kruscal
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1102 题意:这道题实际上和hdu 1242 Rescue 非常相似,改变了输入方式之后, 本题实际上更 ...
- hdu 1102 Constructing Roads(最小生成树 Prim)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1102 Problem Description There are N villages, which ...
- HDU 1102(Constructing Roads)(最小生成树之prim算法)
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1102 Constructing Roads Time Limit: 2000/1000 MS (Ja ...
- hdu 1102 Constructing Roads (Prim算法)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=1102 Constructing Roads Time Limit: 2000/1000 MS (Jav ...
- hdu 1102 Constructing Roads (最小生成树)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1102 Constructing Roads Time Limit: 2000/1000 MS (Jav ...
- HDU 1102 Constructing Roads (最小生成树)
最小生成树模板(嗯……在kuangbin模板里面抄的……) 最小生成树(prim) /** Prim求MST * 耗费矩阵cost[][],标号从0开始,0~n-1 * 返回最小生成树的权值,返回-1 ...
- hdu 1102(最小生成树)
Constructing Roads Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Other ...
随机推荐
- Python连载32-多线程其他属性以及继承Thread类
一.线程常用属性 1.threading.currentThread:返回当前线程变量 2.threading.enumerate:返回一个包含正在运行的线程的list,正在运行的线程指的是线程启动后 ...
- 微信小程序:使用wx.request()请求后台接收不到参数
问题描述: 微信小程序:wx.request()请求后台接收不到参数,我通过wx.request()使用POST方式调用请求,参数传递不到后台 解决方案: Content-Type': 'applic ...
- find命令常用场景
1.查找/var目录下属主为root并且属组为mail的所有文件: find /var -user root -group mail 2.查找/usr目录下不属于root,bin,或student的文 ...
- 解决 “version `GLIBCXX_3.4.21' not found ”问题
https://blog.csdn.net/Heldrecom/article/details/85040411
- 第二节: Redis之Set类型和SortedSet类型的介绍和案例应用
一. Set类型基础 1. 类型说明 1个key→多个value,value的值不重复! Set一种无序且元素内容不重复的集合,不用做重复性判断了,和我们数学中的集合概念相同,可以对多个集合求交集.并 ...
- 【转】Git GUI基本操作
一.Git GUI基本操作 1.版本库初始化 gitpractise文件夹就变成了Git可以管理的仓库,目录下多了一个.git文件夹,此目录是Git用于管理版本库的,不要擅自改动里面的文件,这样会破坏 ...
- C# NuGet常用命令
命令执行位置:工具=〉Nuget包管理器=〉程序包管理器控制台 一.安装 1.安装指定版本类库install-package <程序包名> -version <版本号> 2.安 ...
- Java基础—内部类
在Java语言中,可以把一个类定义到另一个类的内部,在类里面的这个类就叫作内部类,外面的类叫作外部类.在这种情况下,这个内部类可以被看成外部类的是一个成员(与类的属性和方法类似).还有一种类被称为顶层 ...
- 命令 docker rm | docker rmi | docker prune 的差异
区别: docker rm : 删除一个或多个 容器 docker rmi : 删除一个或多个 镜像 docker prune : 用来删除不再使用的 docker 对象 一.docker rm 命令 ...
- libtool编译
1.充分利用共享库的能力.libtool 是一个通用库支持脚本 2.我们可以认为libtool是gcc的一个抽象,也就是说,它包装了gcc或者其他的任何编译器,用户无需知道细节,只要告诉libtool ...