首页 > 代码库 > HDU1875——畅通工程再续(最小生成树:Kruskal算法)

HDU1875——畅通工程再续(最小生成树:Kruskal算法)

畅通工程再续

Description
相信大家都听说一个“百岛湖”的地方吧,百岛湖的居民生活在不同的小岛中,当他们想去其他的小岛时都要通过划小船来实现。现在政府决定大力发展百岛湖,发展首先要解决的问题当然是交通问题,政府决定实现百岛湖的全畅通!经过考察小组RPRush对百岛湖的情况充分了解后,决定在符合条件的小岛间建上桥,所谓符合条件,就是2个小岛之间的距离不能小于10米,也不能大于1000米。当然,为了节省资金,只要求实现任意2个小岛之间有路通即可。其中桥的价格为 100元/米。
Input
输入包括多组数据。输入首先包括一个整数T(T <= 200),代表有T组数据。
每组数据首先是一个整数C(C <= 100),代表小岛的个数,接下来是C组坐标,代表每个小岛的坐标,这些坐标都是 0 <= x, y <= 1000的整数。
Output
每组输入数据输出一行,代表建桥的最小花费,结果保留一位小数。如果无法实现工程以达到全部畅通,输出”oh!”.
Sample Input
2
2
10 10
20 20
3
1 1
2 2
1000 1000
Sample Output
1414.2
oh!

题目大意:

    中文,略。

结题思路:

    简单的最小生成树,用Kruskal算法过的。

    这个题的顶点是用(x,y)存的,可以用结构体存。

    通过判断是否会有N-1条边加入树来判断是否可以生成最小树。

    PS:sqrt()里面要是INT型等整型,需要强制转换,因为math.h中有 double sqrt(double),float sqrt(flaot),long double sqrt(long double)三个函数。HDU会报编译错误。

Code:

 1 #include<stdio.h> 2 #include<string> 3 #include<vector> 4 #include<algorithm> 5 #include<cmath> 6 #include<iostream> 7 #define MAXN 5000 8 using namespace std; 9 struct point10 {11     int x,y;12     int father;13 } P[120];14 struct edge15 {16     int begin;17     int end;18     double dis;19 } T[MAXN+10];20 void init()21 {22     for (int i=1; i<=MAXN; i++)23         P[i].father=i;24 }25 int find(int x)26 {27     if (P[x].father!=x)28         P[x].father=find(P[x].father);29     return P[x].father;30 }31 void join(int x,int y)32 {33     int fx=find(x),fy=find(y);34     if (fx!=fy)35         P[fx].father=fy;36 }37 bool cmp(struct edge a,struct edge b)38 {39     return a.dis<b.dis;40 }41 double dis2(int x1,int y1,int x2,int y2)42 {43     return sqrt((double)((x1-x2)*(x1-x2)+(y1-y2)*(y1-y2)));44 }45 int main()46 {47     int S;48     cin>>S;49     while (S--)50     {51         init();52         int C;53         cin>>C;54         for (int i=1; i<=C; i++)55             cin>>P[i].x>>P[i].y;56         int k=1;57         bool ok=0;58         for (int i=1; i<=C; i++)59             for (int j=1; j<i; j++)60             {61                 T[k].dis=dis2(P[i].x,P[i].y,P[j].x,P[j].y);62                 T[k].begin=i;63                 T[k].end=j;64                 k++;65             }66         sort(T+1,T+k,cmp);67         int sum=0;68         double cnt=0;69 70         for (int i=1; i<k; i++)71         {72             if (find(T[i].begin)!=find(T[i].end)&&73                 (10.000000<=T[i].dis&&T[i].dis<=1000.000001))74             {75                 sum++;76                 cnt+=T[i].dis;77                 join(T[i].begin,T[i].end);78             }79             if (sum==C-1)80             {81                 ok=1;82                 break;83             }84         }85         if (ok) printf("%.1lf\n",cnt*100);86         else printf("oh!\n");87     }88     return 0;89 }