题目转自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 ;
}

今天的讲解就到这了,若果有没有听懂的可以借鉴一下《啊哈!算法》

最新文章

  1. ElasticSearchwindow下搭建
  2. 编写使用SpringSecurity的JUnit测试提醒
  3. Python cookbook-读书笔记01
  4. 基于Qt的P2P局域网聊天及文件传送软件设计
  5. &lt;!DOCTYPE html PUBLIC &quot;-//W3C//DTD XHTML 1.0 Transitional//EN&quot; &quot;http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd&quot;&gt;
  6. HDOJ(HDU) 2109 Fighting for HDU(简单排序比较)
  7. Swing-布局管理器之GridLayout(网格布局)-入门
  8. ORM Basic
  9. 阿里云CentOS7服务器利用LVM分区挂载磁盘全记录
  10. Docker 网络背后的原理探索
  11. [android] 保存文件到SD卡
  12. C语言之指针变量
  13. Asp.net的HttpContext.Current.Items详解
  14. 在Activity中使用Thread导致的内存泄漏
  15. asp.net GridView实现多表头类 多行表头实现方法
  16. Dockerfile的一些demo
  17. C/C++中 malloc和new区别
  18. jQuery插件之ajaxFileUpload[转载]
  19. 时间插件datetimepicker
  20. Hadoop之MapReduce命令

热门文章

  1. POJ-2661Factstone Benchmark
  2. python asyncio call_soon, call_at, call_later
  3. Excel 快捷键
  4. springboot学习源码
  5. 基于OceanStor Dorado V3存储之精简高效 Smart 系列特性
  6. trailhead学习之 LWC for Aura Developers
  7. PHP 数组函数大全
  8. 微信小程序自定义tabbar的实现
  9. Qt 连接MySQL
  10. Qt导航栏 QListWidget