poj1258_prime

<pre name="code" class="cpp">/*
题意:给你一个n*n的连通图,求将他全部连起来的最小花费,即求其最小生成树

这题虽然用kruskal更合适(下面写了原因),但这里我们用prim算法<pre>
1).输入:一个加权连通图,其中顶点集合为V,边集合为E;
2).初始化:Vnew = {x},其中x为集合V中的任一节点(起始点),Enew = {},为空;
3).重复下列操作,直到Vnew = V:
a.在集合E中选取权值最小的边<u, v>,其中u为集合Vnew中的元素,而v不在Vnew集合当中,
并且v∈V(如果存在有多条满足前述条件即具有相同权值的边,则可任意选取其中之一);
b.将v加入集合Vnew中,将<u, v>边加入集合Enew中;
4).输出:使用集合Vnew和Enew来描述所得到的最小生成树。
 
prim算法适合稠密图,其时间复杂度为O(n^2),其时间复杂度与边得数目无关,
而kruskal算法的时间复杂度为O(eloge)跟边的数目有关,适合稀疏图。
*/
#include<stdio.h>
#include<string.h>
#include<limits.h>
#define N 102
//map[][]是给定的带权无向图,low[]记录了每两个点之间当前的最小值,vis[]标记是否被访问过 
int n,map[N][N],low[N],vis[N],ans;
int prime()
{
	memset(vis,0,sizeof vis);//标记都未访问 即 Enew = {} 
	int pos ,min,res=0;//pos记录起点,min保存i到j的最小值,res保存当前最短路 
	//这里假设起点是1
	vis[1]=1;//Vnew = {1} 
	pos=1; 
	//初次给low数组赋值
	for(int i=1;i<=n;i++)
	    if(i!=pos)
		    low[i]=map[pos][i];
    //再运行n-1次 
     for(int i=1;i<n;i++)
	 {
	 	//找出最小权值并记录位置
	 	 min=INT_MAX;
		 for(int j=1;j<=n;j++)
 		      if(!vis[j]&&min>low[j])
			   {
   			       min=low[j];
				   pos=j;	
		       }
       //最小值相加获得最小生成树的
       res+=min;
       //标记该点 
       vis[pos]=1;//Vnew = {1,,,pos}  
       //更新low[]
	   for(int j=1;j<=n;j++)
	       if(!vis[j]&&low[j]>map[pos][j])
		       low[j]=map[pos][j]; 
 	 } 
   return res;
}
int main()
{
    while(scanf("%d",&n)!=EOF)
	{
		//map[][]初始化为 INT_MAX,即认为任何两点都不连接 
		memset(map,INT_MAX,sizeof map);
        for(int i=1;i<=n;i++)
           for(int j=1;j<=n;j++)
                scanf("%d",&map[i][j]);
        ans=prime();
        printf("%d\n",ans);
	}	
}


 

 


                
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值