算法核心思想:
Prim算法代码:
#define INF 32767 //INF表示∞
void Prim(MatGraph g,int v)
{
int lowcost[MAXV]; //顶点i是否在U中
int min;
int closest[MAXV],i,j,k;
for (i=0;i<g.n;i++) //给lowcost[]和closest[]置初值
{
lowcost[i]=g.edges[v][i];
closest[i]=v;
}
for (i=1;i<g.n;i++) //找出n-1个顶点
{
min=INF;
for (j=0;j<g.n;j++) //在(V-U)中找出离U最近的顶点k
if (lowcost[j]!=0 && lowcost[j]<min)
{
min=lowcost[j];
k=j; //k记录最近顶点的编号
}
printf(" 边(%d,%d)权为:%d\n",closest[k],k,min);
lowcost[k]=0; //标记k已经加入U
for (j=0;j<g.n;j++) //修改数组lowcost和closest
if (g.edges[k][j]!=0 && g.edges[k][j]<lowcost[j])
{
lowcost[j]=g.edges[k][j];
closest[j]=k;
}
}
}
简化版理解:
手工实现过程示例:
例题:应试解法
[小结]最小生成树研究的都是带权连通无向图!最小生成树的边数为顶点数减1。最小生成树的代价即权值之和是唯一的,且是最小的,但是其树形不唯一,因为可能存在权值相同的边;如果没有权值相同的边,则最小生成树就是唯一的。所以说,用普里姆算法和克鲁斯卡尔算法构造出图的最小生成树可能相同,也可能不同。