【Gym 100971A】Treasure Island

时间:2021-07-28 04:51:25

题意

题目链接
给你一个地图,'#'代表水,'.'代表陆地,'?'代表擦去的地图,可能是'#'也可能是'.'。地图中本该只有一块相连的陆地,若只有一种方案则输出确定的地图。若有多种方案,则输出‘Ambiguous’,若无答案,则输出‘Impossible’。

分析

将所有‘.’进行dfs扫一遍,dfs时遇到的‘?’当作'.',因为被'#'包围的'?'一定代表‘#’。
一边记录相连的陆地数量b,一边记录当前相连陆地的总格数s0。
如果b==1,就对每个遇到过的'?',将其置为'#‘,再dfs一遍,如果扫到总格数小于s0-1,则有多种答案,否则就是一种答案。
如果b>1,则是无答案。

代码

#include <bits/stdc++.h>
#define N 55
int n, m, a[N][N], u[N][N], x, y, b, s0, s, ok, fx[] = {, , , -}, fy[] = {, -, , };
char c;
void dfs(int x, int y)
{
if (u[x][y]) return;
u[x][y] = ;
s++;
if (a[x][y] == ) a[x][y] = ;
for (int i = ; i < ; i++)
{
int nx = x + fx[i];
int ny = y + fy[i];
if (a[nx][ny])
dfs(nx, ny);
}
}
int main()
{
scanf("%d%d ", & n, & m);
for (int i = ; i <= n; i++)
{
for (int j = ; j <= m; j++)
{
c = getchar();
if (c == '.') a[i][j] = ;
else if (c == '?') a[i][j] = ;
}
getchar();
}
for (int i = ; i <= n; i++)
for (int j = ; j <= m; j++)
if (a[i][j] == && !u[i][j])
{
b++;
dfs(i, j);
x = i;
y = j;
}
if(b == )
{
s0 = s;
for (int i = ; i <= n && !ok; i++)
for (int j = ; j <= m && !ok; j++)
if (a[i][j] == )
{
memset(u, , sizeof u);
s = a[i][j] = ;//? => #
dfs(x, y);
if(s == s0 - )
ok = ;
a[i][j] = ;
}
if (ok)
printf("Ambiguous");
else
for (int i = ; i <= n; i++)
{
for (int j = ; j <= m; j++)
if (a[i][j] == ) printf(".");
else printf("#");
printf("\n");
}
}
else printf("Impossible");
return ;
}