1、题目描述

有一间长方形的房子,地上铺了红色、黑色两种颜色的正方形瓷砖。

你站在其中一块黑色的瓷砖上,只能向相邻(上下左右四个方向)的黑色瓷砖移动。

请写一个程序,计算你总共能够到达多少块黑色的瓷砖。

输入格式

输入包括多个数据集合。

每个数据集合的第一行是两个整数 W 和 H,分别表示 x 方向和 y 方向瓷砖的数量。

在接下来的 H 行中,每行包括 W 个字符。每个字符表示一块瓷砖的颜色,规则如下

1)‘.’:黑色的瓷砖;

2)‘#’:红色的瓷砖;

3)‘@’:黑色的瓷砖,并且你站在这块瓷砖上。该字符在每个数据集合中唯一出现一次。

当在一行中读入的是两个零时,表示输入结束。

输出格式

对每个数据集合,分别输出一行,显示你从初始位置出发能到达的瓷砖数(记数时包括初始位置的瓷砖)。

数据范围

1≤W,H≤20

输入样例:

6 9
....#.
.....#
......
......
......
......
......
#@...#
.#..#.
0 0

输出样例:

45

2、算法描述

本题考察的是Flood fill算法(洪水泛滥算法)、可以借助DFS、BFS两种方式来实现、经典的走迷宫问题、并且可以扩展维度、这里是上下左右四个方向、经常使用到的坐标板子如下所示。

int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};

即借助DFSBFS两种方式、都能解决此类问题、并且套路固定、下面是做法。


1、DFS

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm> using namespace std; const int N = 25; char g[N][N];
int n, m;
bool st[N][N]; // 状态数组 int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0 , -1}; // 坐标
int dfs(int x, int y)
{
int cnt = 1; // 自己从第一块砖出发
st[x][y] = true; for(int i = 0 ; i < 4 ; i ++) // 四个方向
{
int a = x + dx[i], b = y + dy[i];
if(a < 0 || a >= n || b < 0 || b >= m) continue; // 越界
if(g[a][b] == '#') continue; // 踩到红砖
if(st[a][b]) continue; // 已经踩过 cnt += dfs(a, b); // 递归、层级灌溉
} return cnt;
}
int main()
{
while(cin >> m >> n, m || n)
{
for(int i = 0 ; i < n ; i ++) cin >> g[i]; int x, y;
for(int i = 0 ; i < n ; i ++)
for(int j = 0 ; j < m ; j ++ )
if(g[i][j] == '@')
{
x = i;
y = j;
} memset(st, 0, sizeof st); // 每轮都要清空状态
cout << dfs(x, y) << endl;
} return 0;
}

2、BFS

#include <iostream>
#include <cstring>
#include <algorithm>
#include <queue> // BFS需要借助队列 #define x first
#define y second using namespace std; typedef pair<int, int> PII; // 涉及到坐标用pair const int N = 25; char g[N][N];
int m, n;
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1}; int bfs(int sx, int sy)
{
queue<PII> q;
q.push({sx, sy}); //g[sx][sy] = '#'; // 当前踩的点标记
int res = 0;
while(q.size())
{
auto t = q.front(); // 获取队头元素
q.pop(); // 弹出队头元素 res ++ ;
for(int i = 0 ; i < 4 ; i ++) // 拓展四个方向
{
int x = t.x + dx[i], y = t.y + dy[i];
if(x < 0 || x >= n || y < 0 || y >= m || g[x][y] != '.') continue; // 越界情况
g[x][y] = '#'; // 踩过之后标记
q.push({x, y}); // 放入队列
}
}
return res;
}

最新文章

  1. python读取文件夹
  2. wex5 实战 登陆帐号更换与用户id一致性
  3. [MISSAJJ原创]cell内 通过SDWebImage自定义创建动态菊花加载指示器
  4. KnockoutJS 3.X API 第五章 高级应用(1) 创建自定义绑定
  5. Windows Azure Virtual Machine (24) Azure VM支持多网卡功能
  6. Windows Store App 偏移特效
  7. “java.sql.SQLException: Value &#39;0000-00-00&#39; can not be represented as java.sql.Timestamp”
  8. 在centos中使用vim编辑器
  9. mac ssd开启trim
  10. httpd-vhosts.conf的配置
  11. 【原创】leetCodeOj --- Intersection of Two Linked Lists 解题报告(经典的相交链表找交点)
  12. sql server 日期转换函数 convert()
  13. Objective-C处理动态类型函数集
  14. shell编程其实真的很简单(三)
  15. C++临时对象以及针对其进行的优化
  16. Oracle 11g R2创建数据库之手工建库方式
  17. kafka监控kafka-eagle 容器化配置
  18. tp剩余未验证内容-4
  19. HDU 3746 - Cyclic Nacklace &amp; HDU 1358 - Period - [KMP求最小循环节]
  20. 自己一下午练习Js的代码

热门文章

  1. 【2】蛋白鉴定软件之Comet
  2. Linux服务器I/O性能分析-1
  3. 在前端页面中使用Markdown并且优化a标签
  4. 07 MySQL安装图解--Windows版本
  5. linux下定位异常消耗的线程实战分析
  6. android 防止R被混淆,R类反射混淆,找不到资源ID
  7. Linux基础命令---wget下载工具
  8. 石墨文档Websocket百万长连接技术实践
  9. 运维笔记之yum,rpm,挂载,磁盘管理和raid详解
  10. Thymeleaf+layui+jquery复选框回显