적록색약은 빨간색과 초록색을 구분하지 못하는 사람을 의미한다.
먼저 적록색약이 아닌 사람이 본 구역의 개수를 구한 후에 빨간색을 초록색으로 바꾼 후에 적록색약인 사람이 보는 구역의 개수를 출력하면 된다. 첫번째 영역 개수를 출력하고 방문했는지 확인하는 vis 배열만 초기화해주면 쉽게 해결할 수 있다.
#include <bits/stdc++.h>
using namespace std;
#define X first
#define Y second
#define SIZE 102
char board[SIZE][SIZE];
bool vis[SIZE][SIZE];
int dx[4] = { 1,0,-1,0 };
int dy[4] = { 0,1,0,-1 };
int main()
{
ios::sync_with_stdio(0);
cin.tie(0);
int n, tc = 2;
cin >> n;
for (int i = 0; i < n; ++i)
for (int j = 0; j < n; ++j)
cin >> board[i][j];
while (tc--) // 적록색약이 아닌 경우와 적록색약인 경우
{
int cnt = 0; // 영역의 개수
for (int i = 0; i < n; ++i)
{
for (int j = 0; j < n; ++j)
{
if (vis[i][j]) continue;
queue<pair<int, int>> Q;
Q.push({ i,j });
vis[i][j] = 1;
char c = board[i][j]; // 구역 비교하기 위해
cnt++;
while (!Q.empty())
{
auto cur = Q.front(); Q.pop();
for (int dir = 0; dir < 4; ++dir)
{
int nx = cur.X + dx[dir];
int ny = cur.Y + dy[dir];
if (nx < 0 || nx >= n || ny < 0 || ny >= n)continue;
if (vis[nx][ny] || board[nx][ny] != c) continue;
Q.push({ nx,ny });
vis[nx][ny] = 1;
}
}
}
}
cout << cnt << '\n';
fill(vis[0], vis[0] + SIZE * SIZE, 0); // 적록색약인 경우 계산하기 위해 초기화
for (int i = 0; i < n; ++i)
for (int j = 0; j < n; ++j)
if (board[i][j] == 'R') // 빨간색이면
board[i][j] = 'G'; // 초록색으로 변경
}
}