题目描述
分析
每一个岛屿对应一个连通块(四方向连通).
则可以利用深搜, 搜索每一个连通块, 判断每一个连通块是否会被完全淹没
实现
#include <cstdio>
#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 1009;
int n;
int dx[] = {1,0,-1,0}, dy[] = {0,-1,0,1};
char map[N][N];
bool vis[N][N]; // 当前岛屿的(x,y)位置是否被访问过
int ans;
int flag; // 当前岛屿是否被完全淹没
bool in(int x, int y) // 坐标是否合法
{
if(x < 0 || y < 0 || x >= n || y >= n) return false;
return true;
}
bool check(int x, int y) // 将要访问的位置是否是未被访问过的岛屿
{
if(vis[x][y]) return false;
if(map[x][y] == '.') return false;
return true;
}
void dfs(int x, int y) // 搜索当前连通块
{
vis[x][y] = 1;
bool flag2 = 1; // 当前岛屿是否存在不被淹没的位置(四周是'#',则不会淹没')
for(int i=0; i<4; i++)
{
int nx = x + dx[i];
int ny = y + dy[i];
if(in(nx,ny) && map[nx][ny] == '.')
{
flag2 = 0;
break;
}
}
if(flag2) flag = 0;
for(int i=0; i<4; i++)
{
int nx = x + dx[i];
int ny = y + dy[i];
if(in(nx, ny) && check(nx, ny))
{
dfs(nx, ny);
}
}
return;
}
int main()
{
scanf("%d",&n);
for(int i=0; i<n; i++)
{
getchar(); // 注意要正确读入地图, 不然一切白费功夫
for(int j=0; j<n; j++)
{
scanf("%c",&map[i][j]);
}
}
for(int i=0; i<n; i++)
{
for(int j=0; j<n; j++)
{
// 搜索当前连通块
flag = 1;
if(map[i][j] == '#' && !vis[i][j])
{
dfs(i,j);
if(flag) ans++;
}
}
}
cout << ans << endl;
return 0;
}