码迷,mamicode.com
首页 > 编程语言 > 详细

Highways POJ-1751 最小生成树 Prim算法

时间:2019-09-27 23:23:38      阅读:151      评论:0      收藏:0      [点我收藏+]

标签:kruskal   默认   eof   使用   点距   turn   ==   判断   还需   

Highways POJ-1751 最小生成树 Prim算法

题意

有一个N个城市M条路的无向图,给你N个城市的坐标,然后现在该无向图已经有M条边了,问你还需要添加总长为多少的边能使得该无向图连通.输出需要添加边的两端点编号即可。

解题思路

这个可以使用最短路里面的Prim算法来实现,对于已经连接的城市,处理方式是令这两个城市之间的距离等于0即可。

prim算法可以实现我们具体的路径输出,Kruskal算法暂时还不大会。

代码实现

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int inf=0x3f3f3f3f;
const int maxn=1e3;
struct Node{
    int x, y;
}node[maxn];
int mp[maxn][maxn];
int vis[maxn];
int dis[maxn];//这里的dis存储的是其他各个点,到最小生成树中任意一点的最小值。
int line[maxn];
int n, m, ans=0;
void init()
{
    for(int i=1; i<=n; i++)
    {
        dis[i]=mp[i][1];//开始的时候任选1号顶点加入到生成树中。这里是其他点到1号顶点的距离。
        line[i]=1; //默认没有加入到生成树的点距离生成树中最近距离的点是1;
        vis[i]=0; //默认没有点加入到生成树中。
    }
}
bool prim()
{
    ans=0;
    vis[1]=1;
    for(int i=1; i<n; i++)
    {
        int tmp=inf, k;
        for(int j=1; j<=n; j++)
        {
            if(!vis[j] && dis[j]<tmp)
            {
                tmp=dis[j];
                k=j;
            }
        }
        if(tmp==inf) return false;//生成最小树失败,该图不是连通的。
        vis[k]=1;
        ans+=dis[k];
        if(mp[k][line[k]]!=0)//这里判断不为0,是因为有的路已经修好了,就不用输出了
        {
            printf("%d %d\n", k, line[k]); //输出需要链接的两个点
        }
        for(int j=1; j<=n; j++) //以新加入生成树的点作为中间点,看看能优化
        {
            if(!vis[j] && dis[j] > mp[j][k])
            {
                line[j]=k;
                dis[j]=mp[j][k];
            }
        }
    }
    return true;
}
int main()
{
    while(scanf("%d", &n)!=EOF)
    {
        for(int i=1; i<=n; i++)
        {
            scanf("%d%d", &node[i].x, &node[i].y);
        }
        for(int i=1; i<=n; i++)
        {
            for(int j=1; j<=n; j++)
            {
                int tmp=(node[i].x-node[j].x)*(node[i].x-node[j].x)+(node[i].y-node[j].y)*(node[i].y-node[j].y);
                mp[i][j]=tmp;
                mp[j][i]=tmp;
            }
        }
        scanf("%d", &m);
        int x, y;
        for(int i=1; i<=m; i++)
        {
            scanf("%d%d", &x, &y);
            mp[x][y]=mp[y][x]=0;
        }
        init();
        prim();
    }
    return  0;
}

Highways POJ-1751 最小生成树 Prim算法

标签:kruskal   默认   eof   使用   点距   turn   ==   判断   还需   

原文地址:https://www.cnblogs.com/alking1001/p/11600669.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!