난이도 : 골드 4
유형 : 구현, 시뮬레이션
https://www.acmicpc.net/problem/17144
미세먼지를 제거하기 위해 구사과는 공기청정기를 설치하려고 한다. 공기청정기의 성능을 테스트하기 위해 구사과는 집을 크기가 R×C인 격자판으로 나타냈고, 1×1 크기의 칸으로 나눴다. 구사과는 뛰어난 코딩 실력을 이용해 각 칸 (r, c)에 있는 미세먼지의 양을 실시간으로 모니터링하는 시스템을 개발했다. (r, c)는 r행 c열을 의미한다.
공기청정기는 항상 1번 열에 설치되어 있고, 크기는 두 행을 차지한다. 공기청정기가 설치되어 있지 않은 칸에는 미세먼지가 있고, (r, c)에 있는 미세먼지의 양은 Ar,c이다.
1초 동안 아래 적힌 일이 순서대로 일어난다.
다음은 확산의 예시이다.
공기청정기의 바람은 다음과 같은 방향으로 순환한다.
방의 정보가 주어졌을 때, T초가 지난 후 구사과의 방에 남아있는 미세먼지의 양을 구해보자.
첫째 줄에 R, C, T (6 ≤ R, C ≤ 50, 1 ≤ T ≤ 1,000) 가 주어진다.
둘째 줄부터 R개의 줄에 Ar,c (-1 ≤ Ar,c ≤ 1,000)가 주어진다. 공기청정기가 설치된 곳은 Ar,c가 -1이고, 나머지 값은 미세먼지의 양이다. -1은 2번 위아래로 붙어져 있고, 가장 윗 행, 아랫 행과 두 칸이상 떨어져 있다.
첫째 줄에 T초가 지난 후 구사과 방에 남아있는 미세먼지의 양을 출력한다.
미세먼지의 확산은 동시에 확산되기 때문에 따로 2차원 배열을 만들어서 처리해줬다.
package 구현;
import java.io.*;
import java.util.StringTokenizer;
/**
* BOJ_17144_G4_미세먼지 안녕!
* @author mingggkeee
* 구현, 시뮬레이션, 완전탐색
* 공기 청정기는 항상 0번열에 설치, 크기는 두 행 차지
*/
public class BOJ_17144 {
static int R,C,T;
static int[][] map;
static int[][] tempMap;
static int[][] circulator = new int[2][2]; // 공기청정기의 행
static boolean[][] isVisited;
static int[][] dir1 = {{0,1},{-1,0},{0,-1},{1,0}};
static int[][] dir2 = {{0,1},{1,0},{0,-1},{-1,0}};
static int result;
public static void main(String[] args) throws IOException{
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
StringTokenizer st = new StringTokenizer(br.readLine());
R = Integer.parseInt(st.nextToken());
C = Integer.parseInt(st.nextToken());
T = Integer.parseInt(st.nextToken());
int temp = 0;
map = new int[R][C];
for(int r=0; r<R; r++) {
st = new StringTokenizer(br.readLine());
for(int c=0; c<C; c++) {
map[r][c] = Integer.parseInt(st.nextToken());
if(map[r][c] == -1) {
circulator[temp][0] = r;
circulator[temp++][1] = 0;
}
}
}
for(int t=0; t<T; t++) {
dust();
circul();
}
for(int r=0; r<R; r++) {
for(int c=0; c<C; c++) {
result += map[r][c];
}
}
System.out.println(result + 2);
}
static void circul() {
isVisited = new boolean[R][C];
tempMap = new int[R][C];
// 위쪽 청정
int r = circulator[0][0];
int c = circulator[0][1];
tempMap[r][c] = -1;
isVisited[r][c] = true;
int stack = 0;
c++;
while(true) {
int nr = r + dir1[stack%4][0];
int nc = c + dir1[stack%4][1];
if(nr>=0 && nc>=0 && nr<R && nc<C) {
if(map[nr][nc] == -1) {
break;
}
tempMap[nr][nc] = map[r][c];
map[r][c] = 0;
isVisited[nr][nc] = true;
} else {
stack++;
nr = r + dir1[stack%4][0];
nc = c + dir1[stack%4][1];
tempMap[nr][nc] = map[r][c];
map[r][c] = 0;
isVisited[nr][nc] = true;
}
r = nr;
c = nc;
}
// 아래쪽 청정
r = circulator[1][0];
c = circulator[1][1];
tempMap[r][c] = -1;
c++;
stack = 0;
while(true) {
int nr = r + dir2[stack%4][0];
int nc = c + dir2[stack%4][1];
if(nr>=0 && nc>=0 && nr<R && nc<C) {
if(map[nr][nc] == -1) {
map[r][c] = 0;
break;
}
tempMap[nr][nc] = map[r][c];
map[r][c] = 0;
isVisited[nr][nc] = true;
} else {
stack++;
nr = r + dir2[stack%4][0];
nc = c + dir2[stack%4][1];
tempMap[nr][nc] = map[r][c];
map[r][c] = 0;
isVisited[nr][nc] = true;
}
r = nr;
c = nc;
}
for(int i=0; i<R; i++) {
for(int j=0; j<C; j++) {
if(!isVisited[i][j]) {
tempMap[i][j] = map[i][j];
}
}
}
map = tempMap;
}
static void dust() {
tempMap = new int[R][C];
for(int r=0; r<R; r++) {
for(int c=0; c<C; c++) {
int sub = map[r][c] / 5;
for(int i=0; i<4; i++) {
int nr = r + dir1[i][0];
int nc = c + dir1[i][1];
if(nr>=0 && nc>=0 && nr<R && nc<C && map[nr][nc]!=-1) {
tempMap[nr][nc] += sub;
map[r][c] -= sub;
}
}
}
}
for(int r=0; r<R; r++) {
for(int c=0; c<C; c++) {
map[r][c] += tempMap[r][c];
}
}
}
}