uvaLive7303 Aquarium (最小生成树)

题意:给R*C的房间,每个房间被左上-右下或右上-左下的墙分割为两个小房间,将分割移除有一定花费,问使所有小房间联通需要的最小花费

把每个房间分成左右(上下?)两个点,判一判,本来就联通的加零边,一个房间里的两个点间加花费的边,跑Kruskal即可

 1 #include<cstdio>
 2 #include<cstring>
 3 #include<algorithm>
 4
 5 using namespace std;
 6 const int maxN=110;
 7
 8 typedef struct{
 9     int a,b,l;
10 }Edge;
11
12
13 Edge eg[maxN*maxN];
14 bool stage[maxN][maxN];  //true->\ false->/
15 int bcj[maxN*maxN*2];  //(i-1)*C+j
16 int T,R,C;
17
18 bool cmp(Edge a,Edge b){
19     return a.l<b.l;
20 }
21
22 int getf(int i){
23     return bcj[i]==i?i:bcj[i]=getf(bcj[i]);
24 }
25
26 void add(int a,int b){  //a->b
27     bcj[getf(a)]=getf(b);
28 }
29
30 int main(){
31     int t,i,j,k,ind,a,b,num,ans;
32     char cr;
33
34     scanf("%d",&T);
35     for(t=1;t<=T;t++){
36         scanf("%d%d\n",&R,&C);
37         for(i=1;i<=R;i++){
38             for(j=1;j<=C;j++){
39                 scanf("%c",&cr);
40                 stage[i][j]=(cr==‘\\‘);
41             }
42             if(i<R)  scanf("\n");
43         }
44         ind=0;
45         for(i=1;i<=R;i++){
46             for(j=1;j<=C;j++){
47                 eg[ind].a=(i-1)*C*2+j*2-1;
48                 eg[ind].b=(i-1)*C*2+j*2;
49                 scanf("%d",&eg[ind++].l);
50             }
51         }
52         for(i=1;i<=R;i++){
53             ind=(i-1)*C*2;
54             bcj[ind+1]=ind+1;
55             bcj[ind+C*2]=ind+C*2;
56             for(j=1;j<C;j++){
57                 bcj[ind+j*2]=ind+j*2;
58                 bcj[ind+j*2+1]=ind+j*2;
59             }
60         }
61         for(i=1;i<R;i++){
62             for(j=1;j<=C;j++){
63                 ind=(i-1)*C*2+j*2;
64                 a=stage[i][j]?ind-1:ind;
65                 b=stage[i+1][j]?ind+C*2:ind+C*2-1;
66                 add(a,b);
67             }
68         }
69         sort(eg,eg+R*C,cmp);
70
71         ans=0;
72         for(i=0;i<R*C;i++){
73             if(getf(eg[i].a)!=getf(eg[i].b)){
74                 add(eg[i].a,eg[i].b);
75                 ans+=eg[i].l;
76             }
77         }
78         printf("Case %d: %d\n",t,ans);
79
80     }
81 }

原文地址:https://www.cnblogs.com/Ressed/p/9354109.html

时间: 2024-11-05 19:40:54

uvaLive7303 Aquarium (最小生成树)的相关文章

UVALive-7303 Aquarium (最小生成树)

题目大意:在nxm的方格中,每一个1x1的小方格中都有一堵沿对角线的墙,并且每堵墙都有一个坚固程度,这些墙将nxm的方格分割成了若干个区域.现在要拆除一些墙,使其变成一个区域. 题目分析:将区域视作点,将墙视作边,这样问题就变成了求最小生成树. 代码如下: # include<iostream> # include<cstdio> # include<cmath> # include<vector> # include<list> # inclu

次最小生成树 模版

次小生成树(转) 转载(http://www.cnblogs.com/z360/p/6875488.html) 所谓次小生成树,顾名思义就是从生成树中取出的第二小的生成树. 我们在前面已经说过最小生成树的概念及代码实现了,所以接下来要说的次小生成树应该比较简单理解了. 求次小生成树的两种方法 1:首先求出最小生成树T,然后枚举最小生成树上的边,计算除了枚举的当前最小生成树的边以外的所有边形成的最小生成树Ti,然后求最小的Ti就是次小生成树.2:首先计算出最小生成树T,然后对最小生成树上任意不相邻

HDU1863 畅通工程---(最小生成树)

畅通工程 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 27972    Accepted Submission(s): 12279 Problem Description 省政府"畅通工程"的目标是使全省任何两个村庄间都可以实现公路交通(但不一定有直接的公路相连,只要能间接通过公路可达即可).经过调查评估,得到的统计表中列出

51Nod1601 完全图的最小生成树计数

传送门 我居然忘写题解啦!(记忆废) 不管怎么说,这题还算是一道好题啊--你觉得敦爷出的题会有水题么 -- 这题比较容易把人误导到Boruvka算法之类的东西上去(我们机房去刚D题的人一开始大多也被误导了),但仔细思考之后是可以发现问题的特殊性质的. 听说很多人是从Kruskal算法想到这道题的做法的?好吧我并不是,那我就写写我的思考过程好了-- 记得算导上有一道思考题,判断一个最小生成树算法的正确性.那个算法是这样的:把当前图的点集随意划分成两半,递归两半后选出连接两个点集的边中权值最小的一条

最小生成树求法 Prim + Kruskal

prim算法的思路 和dijkstra是一样的 每次选取一个最近的点 然后去向新的节点扩张 注意这里的扩张 不再是 以前求最短路时候的到新的节点的最短距离 而是因为要生成一棵树 所以是要连一根最短的连枝 所以关键部分修改一下 dist[u] = min(dist[u], e.cost) --->>e是连接 v 和 u的边 同样地 普同写法O(v^2) 用队列优化后O(E*logV) 1 #include <iostream> 2 #include <stdio.h> 3

UVALive-7303- Aquarium【最小生成树】【连通块】

UVALive - 7303- Aquarium 题目链接:7303 题目大意:给你一个r * c的格子,每个格子有一个 ' \ ' 或者 '/' 的墙,以及打掉墙的费用,问使得所有块联通的最小费用.(看图好理解) 题目思路:就是将他化成一个图,联通的块相当于一个点,两个点之间有一条边,边的权值为墙的费用. 转化为连通块的思路是:每个格子看成两部分,左侧和右侧.以一行来看,假设两个格子A,B.那么B格子的右侧的编号一定和A格子的左侧的编号相同.如图所示 给每个格子的左右侧标上号,然后加入边,边的

BZOJ_1016_[JSOI2008]_最小生成树计数_(dfs+乘法原理)

描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1016 给出一张图,其中具有相同权值的边的数目不超过10,求最小生成树的个数. 分析 生成树的计数有一个什么什么算法... 我真的企图研究了...但是智商捉急的我实在看不懂论文... 所以最后还是写了暴力... 当然暴力也要靠正确的姿势的. 首先来看一个结论: 同一张图的所有最小生成树中,边权值相同的边的数目是一定的. 也就是说,假如某一张图的某一棵最小生成树由边权值为1,1,2,2,2,3的

poj1861 最小生成树 prim &amp; kruskal

// poj1861 最小生成树 prim & kruskal // // 一个水题,为的只是回味一下模板,日后好有个照应不是 #include <cstdio> #include <algorithm> #include <cstring> #include <vector> #include <iostream> using namespace std; const int MAX_N = 1008; const int INF =

hdu 1875 畅通工程再续(kruskal算法计算最小生成树)

畅通工程再续 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 18411    Accepted Submission(s): 5769 Problem Description 相信大家都听说一个"百岛湖"的地方吧,百岛湖的居民生活在不同的小岛中,当他们想去其他的小岛时都要通过划小船来实现.现在政府决定大力发展百岛湖,发展首先