`
squirrelRao
  • 浏览: 67686 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Prim算法和Kruskal算法

阅读更多

    Prim算法和Kruskal算法都能从连通图找出最小生成树。区别在于Prim算法是挨个找,而Kruskal是先排序再找。

 

    一、Prim算法:

    Prim算法实现的是找出一个有权重连通图中的最小生成树,即:具有最小权重且连接到所有结点的树。(强调的是树,树是没有回路的)。

    Prim算法是这样来做的: 

    首先以一个结点作为最小生成树的初始结点,然后以迭代的方式找出与最小生成树中各结点权重最小边,并加入到最小生成树中。加入之后如果产生回路则跳过这条边,选择下一个结点。当所有结点都加入到最小生成树中之后,就找出了连通图中的最小生成树了。

 

    Prim算法最小生成树查找过程:

 


 

C语言实现:

 

#include <stdio.h>
  #include <stdlib.h>
  #define maxint 1073741824
  int main()
  {
  FILE *input=fopen("input.txt","r"),*out=fopen("output.txt","w");
  int n,m,i,j,x,y,w;
  fscanf(input,"%d %d",&n,&m);
  int map[n][n],E[m][3],tree[m],Mst[n][n];
  /*Mst表示最小生成树的邻接矩阵,map是原图,E是边集,其中E[0]和E[1]是边的两个顶点,E[2]是边的权值,tree是用于判断原图的点是否在最小生成树中*/
  memset(tree,0,sizeof(tree));
  for(i=0; i<n; i++)
  {
  for(j=0; j<n; j++)
  {
  map[i][j]=maxint;
  Mst[i][j]=maxint;
  }
  E[i][0]=E[i][1]=maxint;
  }
  for(i=0; i<m; i++)
  {
  fscanf(input,"%d %d %d",&x,&y,&w);
  if(w<map[x][y])
  {
  map[x][y]=w;
  map[y][x]=w;
  }
  }
  int min=maxint,next=0,now=0,k=0;
  tree[0]=1;
  for(i=0; i<n; i++)
  {
  for(j=0; j<n; j++)
  {
  if(map[now][j]!=maxint && tree[j]==0)
  {
  E[k][0]=now;
  E[k][2]=map[now][j];
  E[k++][1]=j;
  }
  }
  for(j=0; j<k; j++)
  {
  if(E[j][2]<min && tree[E[j][1]]==0)
  {
  min=E[j][2];
  x=E[j][0];
  y=E[j][1];
  next=y;
  }
  }
  tree[next]=1;
  now=next;
  Mst[x][y]=map[x][y];
  Mst[y][x]=map[y][x];
  min=maxint;
  }
  for(i=0; i<n; i++)
  {
  for(j=0; j<n; j++)
  {
  if(Mst[i][j]==maxint) //判断两点是否连通
  fprintf(out,"00 "); //美化输出,不必多加探究
  else
  {
  fprintf(out,"%d ",Mst[i][j]); //输出生成树的邻接矩阵,要输出树的自己可以根据邻接矩阵的数据进行加工
  }
  }
  fprintf(out,"\n");
  }
  fclose(input);
  fclose(out);
  return 0;
  } // 程序未考虑不是连通图的情况,修改很简单,判断生成树的节点数量是否等于原图的节点数量
  //如果小于(不会有大于)则本图不是连通图
  //其实prim和迪杰斯特拉算法核心有相似之处
 

 

    二、Kruskal算法:

    Kruskal算法与Prim算法的不同之处在于,Kruskal在找最小生成树结点之前,需要对所有权重边做从小到大排序。将排序好的权重边依次加入到最小生成树中,如果加入时产生回路就跳过这条边,加入下一条边。当所有结点都加入到最小生成树中之后,就找出了最小生成树。

 

C语言实现:

 

/* Kruskal.c
  Copyright (c) 2002, 2006 by ctu_85
  All Rights Reserved.
  */
  /* I am sorry to say that the situation of unconnected graph is not concerned */
  #include "stdio.h"
  #define maxver 10
  #define maxright 100
  int G[maxver][maxver],record=0,touched[maxver][maxver];
  int circle=0;
  int FindCircle(int,int,int,int);
  int main()
  {
  int path[maxver][2],used[maxver][maxver];
  int i,j,k,t,min=maxright,exsit=0;
  int v1,v2,num,temp,status=0;
  restart:
  printf("Please enter the number of vertex(s) in the graph:\n");
  scanf("%d",&num);
  if(num>maxver||num<0)
  {
  printf("Error!Please reinput!\n");
  goto restart;
  }
  for(j=0;j<num;j++)
  for(k=0;k<num;k++)
  {
  if(j==k)
  {
  G[j][k]=maxright;
  used[j][k]=1;
  touched[j][k]=0;
  }
  else
  if(j<k)
  {
  re:
  printf("Please input the right between vertex %d and vertex %d,if no edge exists please input -1:\n",j+1,k+1);
  scanf("%d",&temp);
  if(temp>=maxright||temp<-1)
  {
  printf("Invalid input!\n");
  goto re;
  }
  if(temp==-1)
  temp=maxright;
  G[j][k]=G[k][j]=temp;
  used[j][k]=used[k][j]=0;
  touched[j][k]=touched[k][j]=0;
  }
  }
  for(j=0;j<num;j++)
  {
  path[j][0]=0;
  path[j][1]=0;
  }
  for(j=0;j<num;j++)
  {
  status=0;
  for(k=0;k<num;k++)
  if(G[j][k]<maxright)
  {
  status=1;
  break;
  }
  if(status==0)
  break;
  }
  for(i=0;i<num-1&&status;i++)
  {
  for(j=0;j<num;j++)
  for(k=0;k<num;k++)
  if(G[j][k]<min&&!used[j][k])
  {
  v1=j;
  v2=k;
  min=G[j][k];
  }
  if(!used[v1][v2])
  {
  used[v1][v2]=1;
  used[v2][v1]=1;
  touched[v1][v2]=1;
  touched[v2][v1]=1;
  path[0]=v1;
  path[1]=v2;
  for(t=0;t<record;t++)
  FindCircle(path[t][0],path[t][0],num,path[t][0]);
  if(circle)
  {/*if a circle exsits,roll back*/
  circle=0;
  i--;
  exsit=0;
  touched[v1][v2]=0;
  touched[v2][v1]=0;
  min=maxright;
  }
  else
  {
  record++;
  min=maxright;
  }
  }
  }
  if(!status)
  printf("We cannot deal with it because the graph is not connected!\n");
  else
  {
  for(i=0;i<num-1;i++)
  printf("Path %d:vertex %d to vertex %d\n",i+1,path[0]+1,path[1]+1);
  }
  return 1;
  }
  int FindCircle(int start,int begin,int times,int pre)
  { /* to judge whether a circle is produced*/
  int i;
  for(i=0;i<times;i++)
  if(touched[begin]==1)
  {
  if(i==start&&pre!=start)
  {
  circle=1;
  return 1;
  break;
  }
  else
  if(pre!=i)
  FindCircle(start,i,times,begin);
  else
  continue;
  }
  return 1;
  }
 

 无疑,Kruskal算法在效率上要比Prim算法快,因为Kruskal只需要对权重边做一次排序,而Prim算法则需要做多次排序。尽管Prim算法每次做的算法涉及的权重边不一定会涵盖连通图中的所有边,但是随着所使用的排序算法的效率的提高,Kruskal算法和Prim算法之间的差异将会清晰的显性出来。

  • 大小: 55.5 KB
1
3
分享到:
评论
1 楼 最佳蜗牛 2013-07-07  
赞!如果Kruskal算法有图示会更好的。 

相关推荐

    Prim算法和Kruskal算法的Matlab实现[归纳].pdf

    Prim算法和Kruskal算法的Matlab实现 Prim算法和Kruskal算法是两种常用的最小生成树算法,它们在计算机仿真和软件开发中有广泛的应用。本文将对这两种算法进行深入的分析和比较,并使用Matlab实现它们的代码实现。 ...

    Prim算法与Kruskal算法求最小生成树

    在这个问题中,Prim算法和Kruskal算法是两种常用的方法。 1. Prim算法: Prim算法是一种贪心策略,它从一个起始顶点开始,逐步添加边,每次添加一条与已选择顶点集形成最小权重边的新边。这个过程不断进行,直到...

    最小生成树 prim算法和Kruskal算法实现

    prim算法 Kruskal算法分别实现最小生成树

    prim算法和kruskal算法

    Prim算法和Kruskal算法 Prim算法是解决最小生成树问题的一种贪心算法,它使用贪心策略来选择当前最优的边添加到生成树中。该算法的思路是从任意一个顶点开始,每次选择与当前顶点最近的一个顶点,并将两点之间的边...

    cpp-图论算法最小生成树Prim算法和Kruskal算法C实现

    本文将详细介绍两种经典算法来解决最小生成树问题:Prim算法和Kruskal算法,并以C语言实现为例进行讲解。 ### Prim算法 Prim算法是一种贪心算法,其基本思想是从一个初始节点开始,逐步添加最短的边,直到连接到...

    Prim算法和Kruskal算法的Matlab实现.pdf

    Prim算法和Kruskal算法的Matlab实现 本文主要介绍了Prim算法和Kruskal算法的Matlab实现,旨在解决连通图的最小支撑树问题。文章首先介绍了连通图的定义和最小支撑树的概念,然后详细介绍了Prim算法和Kruskal算法的...

    用Prim和Kruskal算法构造最小生成树

    本示例通过使用Prim算法和Kruskal算法,展示了如何在一个给定的图中找到最小生成树。Prim算法适用于处理稀疏图,而Kruskal算法则适用于稠密图。这两种算法都是通过贪心策略来构建最小生成树,最终得到的最小生成树将...

    Prim算法和Kruskal算法的Matlab实现

    ### Prim算法和Kruskal算法的Matlab实现详解 #### Prim算法详解 Prim算法是一种用于在加权连通图中寻找最小生成树的算法。它从一个任意的顶点开始,逐步选择权重最小的边来扩展已形成的树,直到包含所有的顶点。...

    prim和kruskal算法求最小生成树

    Prim算法和Kruskal算法各有特点。Prim算法适合稠密图,因为它每次迭代都考虑了所有相邻顶点,而Kruskal算法则更适合稀疏图,因为它的主要时间消耗在于边的排序。在实现上,Prim算法通常使用优先队列优化查找最小边的...

    Prim算法和Kruskal算法的Matlab实现.docx

    Prim算法和Kruskal算法是两种经典的图论算法,主要用于寻找加权无向图中的最小生成树。最小生成树是一棵树形子图,包含了原图的所有顶点,且边的权重之和最小。 **Prim算法**的主要思想是从一个起始顶点开始,逐步...

    c语言实现最小生成树的prim算法和kruskal算法

    详细的c语言实现最小生成树的prim算法和kruskal算法,非常有用的

    Prim算法和Kruskal算法的Matlab实现.doc

    Prim算法和Kruskal算法的Matlab实现.doc

    数据结构 图的最小生成树 C++描述 使用prim算法、kruskal算法

    我们将探讨两种经典的算法:Prim算法和Kruskal算法,它们都是用C++实现的。 1. **最小生成树**: - 最小生成树是图论中的一个重要概念,适用于有向或无向加权图。在一个加权图中,最小生成树是一组边,它们连接了...

    Python实现最小生成树:Prim算法与Kruskal算法详解

    通过上述代码示例,我们可以看到如何在Python中实现Prim算法和Kruskal算法来构建最小生成树。这些技术在实际的软件开发和数据处理中有着广泛的应用,尤其是在需要优化网络连接和降低成本的场景中。随着技术的发展,...

    Prim算法和Kruskal算法Dijkstra算法.zip

    Prim算法通常使用邻接矩阵或邻接表来表示图,Kruskal算法则更依赖于边的排序,而Dijkstra算法通常结合了优先队列(如二叉堆)和邻接表。在实现过程中,需要注意效率优化,例如使用 Fibonacci堆以改善Dijkstra算法的...

    Prim与Kruskal算法的最小生成树matlab实现

    Prim算法和Kruskal算法是两种常用的求解最小生成树的方法,它们各有特点和适用场景。 1. **Prim算法**: Prim算法是从一个顶点开始,逐步添加边来构建最小生成树。它每次选择与当前树连接且权重最小的边,直到所有...

    python最小生成树-Prim算法和Kruskal算法.docx

    prim算法求最小生成树

    Matlab版prim Kruskal算法实现文档

    Matlab版Prim Kruskal算法实现文档 ...Matlab版Prim Kruskal算法实现文档提供了一种使用Matlab实现Prim算法和Kruskal算法的方法,通过实验步骤,可以对比两种算法的实现复杂度和效率,并根据实际情况选择合适的算法。

    C++ Prim算法Kruskal算法构造可以使n个城市连接的最小生成树

    (1)、实验题目:给定一个地区的n 个城市间的距离网,用Prim算法或Kruskal算法建立最小生成树,并得到的最小生成树的代价。 (2)、实验要求: 1、城市间的距离网采用的邻接矩阵表示,邻接矩阵的存储结构定义采用...

    C++实现图的存储、Prim和Kruskal算法

    本项目将深入探讨如何用C++通过邻接矩阵来存储图,并实现两种经典算法:Prim算法和Kruskal算法,用于构建最小生成树。 首先,我们来看一下邻接矩阵的存储方式。邻接矩阵是一个二维数组,数组的大小通常是n×n,其中...

Global site tag (gtag.js) - Google Analytics