最小生成树-水题-POJ-1789-Truck History

题目大意 读入有N个7位的字符串,定义每个字符串之间的距离是他们之间不同字符的个数,然后就是求最小生成树的水题了

//下面用的是算法竞赛入门经典的代码套的,应该好懂吧。。

另外亦可用prim算法来写这题见我的博客点击打开链接还有另一篇点击打开链接

#include<iostream>
#include<algorithm>
#include<cstdio>
#define maxn 2100000
using namespace std;
int u[maxn],v[maxn],w[maxn],xu[maxn],p[2100],m,e;//点的个数,边的个数
char tu[2105][8];
int cal(int i,int j)//计算2个子串的距离的小函数
{
	int ans=0,ii=0;
	while(ii<7)
	{
		if(tu[i][ii]!=tu[j][ii])
			ans++;
			ii++;
	}
	return ans;
}
void read()//读入函数
{
	int ii=e=0;
	for(int i=0;i<m;i++)
			scanf("%s",tu[i]);
	for(int i=0;i<m;i++)
		for(int j=i+1;j<m;j++)
		{
			u[e]=i;
			xu[e]=e;
			v[e]=j;
			w[e++]=cal(i,j);
		}
}
int findp(int x){return p[x]==x?x:p[x]=findp(p[x]);}//并查集的find的函数同时可以不断修改父节点到根
int cmp(const int i,const int j){return w[i]<w[j];}
void deal()//krustra求最短路
{
	sort(xu,xu+e,cmp);
	for(int i=0;i<m+5;i++)
		p[i]=i;
	int ans=0;
	for(int i=0;i<e;i++)
	{
		int ee=xu[i];
		int x=findp(u[ee]),y=findp(v[ee]);
		if(x!=y) {ans+=w[ee]; p[x]=y;}
	}
	printf("The highest possible quality is 1/%d.\n",ans);
}
int main()
{
	while(cin>>m,m)
	{
		read();
		deal();
	}
	return 0;
}




你可能感兴趣的:(最小生成树-水题-POJ-1789-Truck History)