数据结构实验之图论六:村村通公路

数据结构实验之图论六:村村通公路

Time Limit: 1000MS Memory limit: 65536K

题目描述

当前农村公路建设正如火如荼的展开,某乡镇政府决定实现村村通公路,工程师现有各个村落之间的原始道路统计数据表,表中列出了各村之间可以建设公路的若干条道路的成本,你的任务是根据给出的数据表,求使得每个村都有公路连通所需要的最低成本。

输入

连续多组数据输入,每组数据包括村落数目N(N <= 1000)和可供选择的道路数目M(M <= 3000),随后M行对应M条道路,每行给出3个正整数,分别是该条道路直接连通的两个村庄的编号和修建该道路的预算成本,村庄从1~N编号。 

输出

输出使每个村庄都有公路连通所需要的最低成本,如果输入数据不能使所有村庄畅通,则输出-1,表示有些村庄之间没有路连通。 

示例输入

5 8
1 2 12
1 3 9
1 4 11
1 5 3
2 3 6
2 4 9
3 4 4
4 5 6

示例输出

19

 

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_VERTEX_NUM 100	// 最大顶点个数
#define INFINITY 100000	// 用整型最大值代替∞
typedef int VRType;;
typedef int VertexType;
VRType sum,n,m,visited[1000];
// 邻接矩阵的数据结构
typedef struct
{
	VRType adj; // 顶点关系类型。对无权图,用1(是)或0(否)表示相邻否;
				// 对带权图,则为权值类型
 }ArcCell, AdjMatrix[MAX_VERTEX_NUM][MAX_VERTEX_NUM];

// 图的数据结构
typedef struct
{
	VertexType vexs[MAX_VERTEX_NUM];	// 顶点向量
	AdjMatrix arcs;		// 邻接矩阵
	int vexnum,			// 图的当前顶点数
		arcnum;			// 图的当前弧数
} MGraph;

// 记录从顶点集U到V-U的代价最小的边的辅助数组定义
typedef struct
{
	VertexType adjvex;
	VRType lowcost;
}minside[MAX_VERTEX_NUM];
// 算法7.2  P162
// 采用数组(邻接矩阵)表示法,构造无向网G。
int CreateAN(MGraph *G)
{
	int i,j,k,w,v1,v2;
	(*G).vexnum=n;(*G).arcnum=m;
	for(i=1;i<=(*G).vexnum;++i) // 初始化邻接矩阵
		for(j=1;j<=(*G).vexnum;++j)
		{
			(*G).arcs[i][j].adj=INFINITY; // 网初始化为无穷大
		}
	for(k=1;k<=(*G).arcnum;++k)
	{
		scanf("%d%d%d",&v1,&v2,&w);
		if(w<(*G).arcs[v1][v2].adj)
		(*G).arcs[v1][v2].adj=(*G).arcs[v2][v1].adj=w; // 无向
	}
	return 1;
}


// 求closedge.lowcost的最小正值
int minimum(minside SZ,MGraph G)
{
	int i=1,j,k,min;
	while(!SZ[i].lowcost)
		i++;
	min=SZ[i].lowcost; // 第一个不为0的值
	k=i;
	for(j=i+1;j<=G.vexnum;j++)
		if(SZ[j].lowcost>0)
			if(min>SZ[j].lowcost)
			{
				min=SZ[j].lowcost;
				k=j;
			}
	return k;
}

// 算法7.9 P175
// 用普里姆算法从第u个顶点出发构造网G的最小生成树T,输出T的各条边
void MiniSpanTree_PRIM(MGraph G,VertexType u)
{
	int i,j,k;
	minside closedge;
	k=u;
	visited[k]=1;
	for(j=1;j<=G.vexnum;++j) // 辅助数组初始化
	{
		if(j!=k)
		{
			closedge[j].adjvex=u;
			closedge[j].lowcost=G.arcs[k][j].adj;
		}
	}
	closedge[k].lowcost=0; // 初始,U={u}
	for(i=2;i<=G.vexnum;++i)
	{ // 选择其余G.vexnum-1个顶点
		k=minimum(closedge,G); // 求出T的下一个结点:第K顶点
		if(closedge[k].lowcost!=INFINITY)
		visited[k]=1;
		sum=sum+closedge[k].lowcost;
		closedge[k].lowcost=0; // 第K顶点并入U集
		for(j=1;j<=G.vexnum;++j)
			if(G.arcs[k][j].adj<closedge[j].lowcost)
			{
				// 新顶点并入U集后重新选择最小边
				closedge[j].adjvex=k;
				closedge[j].lowcost=G.arcs[k][j].adj;
			}
	}
}

int main()
{
    int i;
    while(scanf("%d%d",&n,&m)!=EOF)
    {
    memset(visited,0,sizeof(visited));
    sum=0;
	MGraph G;
	CreateAN(&G);
	MiniSpanTree_PRIM(G,1);
	for(i=1;i<=n;i++)
        if(visited[i]==0)
        break;
    if(i==n+1)
	printf("%d\n",sum);
	else
        printf("-1\n");
    }
	return 0;
}


 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

猿憨憨

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值