Farm Irrigation

**Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)

Total Submission(s): 4991 Accepted Submission(s): 2143

**

Problem Description

Benny has a spacious farm land to irrigate. The farm land is a rectangle, and is divided into a lot of samll squares. Water pipes are placed in these squares. Different square has a different type of pipe. There are 11 types of pipes, which is marked from A to K, as Figure 1 shows.



Figure 1

Benny has a map of his farm, which is an array of marks denoting the distribution of water pipes over the whole farm. For example, if he has a map

ADC

FJK

IHE

then the water pipes are distributed like



Figure 2

Several wellsprings are found in the center of some squares, so water can flow along the pipes from one square to another. If water flow crosses one square, the whole farm land in this square is irrigated and will have a good harvest in autumn.

Now Benny wants to know at least how many wellsprings should be found to have the whole farm land irrigated. Can you help him?

Note: In the above example, at least 3 wellsprings are needed, as those red points in Figure 2 show.

Input

There are several test cases! In each test case, the first line contains 2 integers M and N, then M lines follow. In each of these lines, there are N characters, in the range of 'A' to 'K', denoting the type of water pipe over the corresponding square. A negative M or N denotes the end of input, else you can assume 1 <= M, N <= 50.

Output

For each test case, output in one line the least number of wellsprings needed.

Sample Input

2 2 DK HF 3 3 ADC FJK IHE -1 -1

Sample Output

2 3

题目大意:给定农田的水管的走向,如果两块农田有水管能够互相连通,则它们是相连的,水流能通过两块农田。要你求出最少需要挖多少口井(水井在每块农田的正中央),才能使所有农田都被灌溉。根据上图例子,需要3口水井就能将所有农田都被灌溉。

分析:这道题有两种方法可以做,第一种是简单dfs,第二种是并查集,dfs如果不懂的同学就要去普及搜索知识了。并查集的话同样,求连通区域的个数,如果两块农田连通,则它们在一个等价类中,最后求等价类个数。

dfs方法实现如下

首先要对11个农田状态进行标记,个人标记的方法各不相同,可以使用二维数组存储,这里我用结构体表示更为直观:

#include<iostream>
#include<cstring>
using namespace std;
struct farm {
bool top, bottom, left, right;
farm() {
top = bottom = left = right = false;
}
}FM[13]; bool book[55][55];
char map[55][55];
int n, m; void init() {
FM[0].left = FM[0].top = true;
FM[1].right = FM[1].top = true;
FM[2].left = FM[2].bottom = true;
FM[3].right = FM[3].bottom = true;
FM[4].top = FM[4].bottom = true;
FM[5].left = FM[5].right = true;
FM[6].left = FM[6].right = FM[6].top = true;
FM[7].left = FM[7].top = FM[7].bottom = true;
FM[8].left = FM[8].right = FM[8].bottom = true;
FM[9].top = FM[9].right = FM[9].bottom = true;
FM[10].left = FM[10].right = FM[10].top = FM[10].bottom = true;
} void dfs(int x, int y)
{
book[x][y] = true;
int c = map[x][y] - 'A';
if (x - 1 >= 0 && FM[c].top && FM[map[x - 1][y] - 'A'].bottom && !book[x - 1][y])
dfs(x - 1, y);
if (y - 1 >= 0 && FM[c].left && FM[map[x][y - 1] - 'A'].right && !book[x][y - 1])
dfs(x, y - 1);
if (x + 1 < m && FM[c].bottom && FM[map[x + 1][y] - 'A'].top && !book[x + 1][y])
dfs(x + 1, y);
if (y + 1 < n && FM[c].right && FM[map[x][y + 1] - 'A'].left && !book[x][y + 1])
dfs(x, y + 1);
} int main() {
int sum;
init();
while (cin >> m >> n) {
if (m < 0 || n < 0) break;
for (int i = 0; i < m; ++i)
for (int j = 0; j < n; ++j)
cin >> map[i][j];
memset(book, false, sizeof(book));
sum = 0;
for (int i = 0; i < m; ++i)
for (int j = 0; j < n; ++j)
if (!book[i][j]) {
++sum;
dfs(i, j);
}
cout << sum << endl;
}
return 0;
}

并查集方法

将二维坐标转化为一维,对每块农田找左、上连通情况,合并等价类

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std; int F[3100];
char mp[55][55];
int m, n;
char pipe[11][5] = {"1100", "0110", "1001", "0011", "0101",
"1010", "1110", "1101", "1011", "0111", "1111"};
int Find(int x)
{
if(F[x] == -1) return x;
return Find(F[x]);
}
void Union(int x, int y)
{
int t1 = Find(x);
int t2 = Find(y);
if(t1 != t2)
F[t1] = t2;
}
int main()
{
while(scanf("%d%d", &m, &n))
{
if(m<0 || n<0) break;
for(int i = 0; i < m; i++)
scanf("%s", &mp[i]);
memset(F, -1, sizeof(F));
for(int i = 0; i < m; i++)
for(int j = 0; j < n; j++)
{
if(i>0 && pipe[mp[i][j]-'A'][1]=='1' && pipe[mp[i-1][j]-'A'][3]=='1')
Union(i*n+j, (i-1)*n+j);
if(j>0 && pipe[mp[i][j]-'A'][0]=='1' && pipe[mp[i][j-1]-'A'][2]=='1')
Union(i*n+j, i*n+j-1);
}
int cnt = 0;
for(int i = 0; i < m*n; i++)
if(F[i] == -1)
cnt++;
printf("%d\n", cnt);
}
return 0;
}

#1198:Farm Irrigation(DFS + 并查集)的更多相关文章

  1. HDU 1198 Farm Irrigation(并查集,自己构造连通条件或者dfs)

    Farm Irrigation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)T ...

  2. hdu 1198 Farm Irrigation(并查集)

    题意: Benny has a spacious farm land to irrigate. The farm land is a rectangle, and is divided into a ...

  3. HDU 1198 Farm Irrigation(并查集+位运算)

    Farm Irrigation Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 65536/32768K (Java/Other) Tot ...

  4. HDU 1198 Farm Irrigation (并查集优化,构图)

    本题和HDU畅通project类似.仅仅只是畅通project给出了数的连通关系, 而此题须要自己推断连通关系,即两个水管能否够连接到一起,也是本题的难点所在. 记录状态.不断combine(),注意 ...

  5. hdu.1198.Farm Irrigation(dfs +放大建图)

    Farm Irrigation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) ...

  6. hdu 1198 Farm Irrigation(深搜dfs || 并查集)

    转载请注明出处:viewmode=contents">http://blog.csdn.net/u012860063?viewmode=contents 题目链接:http://acm ...

  7. hdu1198 Farm Irrigation —— dfs or 并查集

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1198 dfs: #include<cstdio>//hdu1198 dfs #includ ...

  8. 杭电OJ——1198 Farm Irrigation (并查集)

    畅通工程 Problem Description 某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇.省政府"畅通工程"的目标是使全省任何两个城镇间都可 ...

  9. HDU 1198 Farm Irrigation(状态压缩+DFS)

    题目网址:http://acm.hdu.edu.cn/showproblem.php?pid=1198 题目: Farm Irrigation Time Limit: 2000/1000 MS (Ja ...

  10. HDU 1198 Farm Irrigation (并检查集合 和 dfs两种实现)

    Farm Irrigation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) ...

随机推荐

  1. Echarts 饼图,legend样式美化

    最后样式图: 实现代码: var myChart = echarts.init(document.getElementById('container')); let option = { /*{b}: ...

  2. Flask 使用Jinja2模板引擎

    Jinja2,由Flask框架的创作者开发,是一款功能丰富的模板引擎,以其完整的Unicode支持.灵活性.高效性和安全性而备受推崇.最初受Django模板引擎启发,Jinja2为Flask提供了强大 ...

  3. 不会这5个Excel函数,别说你会做数据分析?

    当涉及数据分析时,Excel是一个非常有用的工具,而掌握一些核心函数将大大提高你在数据处理和分析方面的能力.以下是我对五个重要的Excel函数的详细介绍: 1. VLOOKUP 函数 VLOOKUP ...

  4. SpringBoot设置日志级别

    输出到控制台 logging: # 日志记录到文件中 file: # 指定文件名 name: server.log logback: rollingpolicy: # 指定文件大小 max-file- ...

  5. 自定义线程池将异常"吃了"

    今天在做项目时,写了一个使用自定义线程池执行远程调用 // 删除购物车信息 corePoolExecutor.submit(() -> { try { cartFeignClient.delet ...

  6. SpringBoot核心注解:@SpringBootApplication

    @SpringBootApplication它是由三个注解的复合: @ComponentScan @SpringConguration @EnableAutoConfiguration 三个注解的作用 ...

  7. 组合式api-侦听器watch的语法

    和vue2对比,也是语法上稍有不同. 监听单个数据对象 <script setup> import {ref, watch} from "vue"; const cou ...

  8. 【笔记整理】使用Session会话保持

    import requests if __name__ == '__main__': # Session对象实现了客户端和服务器端的每次会话保持功能. session = requests.Sessi ...

  9. 万界星空科技仓库管理wms系统

    ​ 企业在管理库存时,尤其是生产制造企业,使用传统方式比如纸笔.Excel 管理库存,由于工具和信息化存在局限,导致在管理库存时出现如下问题: 1.通过纸笔记录出入库申请,人为手动计算易出错,数据易丢 ...

  10. .Net人的自我修养-书目汇总

    .Net人的自我修养-书目汇总 2019年来现在的公司开始从事.Net相关的工作.记录一下工作以来看过(或者翻过)和听过还不错打算看相关的技术书籍或资料. 为了方便给大家参考,看(翻)过书目以难度排序 ...