poj1979 深度遍历

问题重述
问题描述:
有一个长方形的房间里,是用方瓦覆盖的。每个方瓦的颜色是红色或黑色。一名男子正站在一个黑色瓷砖。他从他所站的方瓦上,可以转移到相邻的四个砖之一。但他无法进入红瓦,他只可以进入黑瓦。
编写一个程序来计算黑瓦数量,也就是他可以达到的方瓦数(重复上述动作)。输入:
输入包含多个数据集。一个数据集的包含有在开始的第一行的两个正整数W和H,W和H表示x和y方向上的方瓦数。 W和H的不超过20。之后有H行的数据集,其中每行包括W列字符。每个字符代表着瓷砖的颜色如下:
’.’ -黑色瓦片
‘#’-一个红色的瓷砖
‘@’-一黑砖上的人
该程序可以循环输入,当输入的W和H都为0时,程序终止。
输出:
有多少块黑瓦
Sample Input
6 9
....#.
.....#
......
......
......
......
......
#@...#
.#..#.
11 9
.#.........
.#.#######.
.#.#.....#.
.#.#.###.#.
.#.#..@#.#.
.#.#####.#.
.#.......#.
.#########.
...........
11 6
..#..#..#..
..#..#..#..
..#..#..###
..#..#..#@.
..#..#..#..
..#..#..#..
7 7
..#.#..
..#.#..
###.###
...@...
###.###
..#.#..
..#.#..
0 0
Sample Output
45
59
6
13
解题过程:
分析过程:
当读懂了题意后,很容易地就地发现这是一道典型的深度遍历的应用。所以在算法上就没有什么问题。当前要解决的就是数据的输入与存储处理问题。一看到矩阵的输入,我就想到了要用二维数组,然后对应的把字符转换为数字存储。考虑到在遍历过程中要判断边界问题,所以我就把数组的行列数再多加两行两列,以方便处理。
编程过程:
在数据的输入方面,我花了较长的时间,我刚开始是用scanf("%s",s);想把每一行的字符读取后,再去掉空格,最后再存进数组。但却发现是这种方式是不读取空格的。之后又想用gets(s);但发现去空格有点麻烦。最后才发现用getchar();就可以很简单地实现。
在数据的存储方面,我用二维数组存储对应的数字,并设置数组初始值为0;因为边界问题,我直接空出一圈不存储数据,使其为默认值,使得边界的处理变很很简单。
在调试的过程中,我犯了一些很低级的错误,花费了很多的时间。其一,break;只跳出当前循环,但我却在一个二层循环中直接使用,以为能跳出两层。其二,因为是循环输入与输出,需要每一趟都初始化。其三,遍历的时候是四个方向逐个访问,但我却因为顺手就用了else if,使得只访问一个方向。
做题收获:
此次编程,让我对字符串的输入处理更熟练,对二维数组做为函数参数的理解更深刻,对广度遍历的算法也更进一步掌握。
4.程序源码:
package middle;


import java.io.BufferedInputStream;
import java.util.Scanner;

/**
*
*poj1979
* 这道题是数据结构的第一道,刚开始是用c语言做的,但后来其他题目我都用了java去写,
* 所以现在只是简单把c语法转化为java法,统一一下编码而已
* @author NC
*/
public class Poj1979 {

private final static int MAXCOL = 22;
private final static int MAXROW = 22;
private static int[][] floor = new int[MAXROW][MAXCOL]; //最多是20行,20列,再加两行两列作为边界
private static int[][] visited = new int[MAXROW][MAXCOL]; //标记是否访问过,默认没有访问过

public static void main(String[] args) {

Scanner scan = new Scanner(new BufferedInputStream(System.in));
while (scan.hasNext()) {
int col = scan.nextInt();
int row = scan.nextInt();
int i = 0, j = 0, k = 0;
int count = 0;
int flag = 0;
char c;
//读取数据
if (col == 0 && row == 0) {
break;
}
//因为是循环输入,所以每一次都得初始化
for (i = 0; i < MAXROW; i++) {
for (j = 0; j < MAXCOL; j++) {
floor[i][j] = 0;
visited[i][j] = 0;
}
}
scan.nextLine();
//一个一个读取字符,将符号转换为数字
for (i = 1; i <= row; i++) {
char[] ss = scan.nextLine().trim().toCharArray();
for (j = 1; j <= col; j++) {
while (true) {
c = ss[j-1];
if (c == '.' || c == '#' || c == '@') {
break;
}
}
if (c == '.') {
floor[i][j] = 1;
} else if (c == '#') {
floor[i][j] = 2;
} else if (c == '@') {
floor[i][j] = 3;
}
}
}
//找出人所站的位置
flag = 0;
for (i = 1; i <= row; i++) {
for (j = 1; j <= col; j++) {
if (floor[i][j] == 3) {
flag = 1;
break; //要跳出两层
}
}
if (flag == 1) {
break;
}
}
//深度遍历
if (floor[i][j] == 3) {
DFS(i, j);
}
//count the number of tiles he can reach from the initial tile
count = 0;
for (i = 1; i <= row; i++) {
for (j = 1; j <= col; j++) {
if (visited[i][j] == 1) {
count++;
}
}
}
//print the number of tiles he can reach from the initial tile
System.out.println(count);
}
}

static void DFS(int i, int j) {
//标记访问过
visited[i][j] = 1;
//四个方向都要遍历,不能用else if
if (floor[i][j + 1] == 1 && visited[i][j + 1] == 0) {
DFS(i, j + 1);
}
if (floor[i][j - 1] == 1 && visited[i][j - 1] == 0) {
DFS(i, j - 1);
}
if (floor[i + 1][j] == 1 && visited[i + 1][j] == 0) {
DFS(i + 1, j);
}
if (floor[i - 1][j] == 1 && visited[i - 1][j] == 0) {
DFS(i - 1, j);
}
}
}


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值