알고리즘

BOJ 23289 - 온풍기 안녕(Java)

장중앙 2022. 1. 12. 13:11

https://www.acmicpc.net/problem/23289

 

23289번: 온풍기 안녕!

유난히 추운 날씨가 예상되는 이번 겨울을 대비하기 위해 구사과는 온풍기를 설치하려고 한다. 온풍기의 성능을 테스트하기 위해 구사과는 집을 크기가 R×C인 격자판으로 나타냈고, 1×1 크기

www.acmicpc.net

 

풀이

조사하는 칸(배열 입력 칸이 5인 칸)이 모두 K 이상일 때까지 1~3 반복후, 초콜릿의 갯수를 출력

1. 모든 온풍기에서 바람이 나옴

  • 온풍기의 방향에 따라 온도가 확산 
    • 하나의 좌표에서 온풍기의 방향에 따라 3방향으로 확산

    • 중첩되는 좌표의 경우 한번만 계산

2. 온도 조절

  • 상하좌우 인접한 칸을 조사, 온도가 높은 칸은 온도를 줄이고 낮은 칸은 온도를 높임
  • 이는 모든 칸에서 한번에 실행됨 

3. 초콜릿 += 1


온풍기의 방향 및 확산 방향

온풍기의 방향에 대해서는 1~4로 입력 값이 주어짐

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

또한, 이러한 온풍기의 방향에 맞게 확산 방향을 지정

static int[][] spreadY = {{0, 0, 0}, {0, -1, 1}, {0, -1, 1}, {-1, -1, -1}, {1, 1, 1}};
static int[][] spreadX = {{0, 0, 0}, {1, 1, 1}, {-1, -1, -1}, {0, -1, 1}, {0, -1, 1}};

 

온풍기 작동

온풍기의 방향에 따른 확산 방향/좌표를 계산

bfs를 이용 너비가 5까지만 탐색하도록 함

온도 상승에 따른 중첩을 계산하지 않기 위해 visit[][]을 사용

격자 범위 밖, 방문한 좌표, 다음 칸에 대해 벽에 막혀있는지를 확인 후, Queue에 삽입하면서 plus[][]에 온도 상승량 저장

private static void work(int i, int j, int dir) {
	boolean[][] visit = new boolean[N][M];
	Queue<Pos> q = new LinkedList<>();

	int t = 5;
	int ny = i + dy[dir];
	int nx = j + dx[dir];

	visit[ny][nx] = true;
	plus[ny][nx] += 5;
	q.add(new Pos(ny, nx, 2));

	while (!q.isEmpty()) {
		Pos now = q.poll();
		int y = now.y;
		int x = now.x;
		int dis = now.dir;

		if (dis > 5) continue;

		for (int d = 0; d < 3; d++) {
			ny = y + spreadY[dir][d];
			nx = x + spreadX[dir][d];
			if (!isRange(ny, nx)) continue;
			if (visit[ny][nx]) continue;
			if (isWall(y, x, ny, nx, dir)) continue;
			visit[ny][nx] = true;
			plus[ny][nx] += t - dis + 1;
			q.add(new Pos(ny, nx, dis + 1));
		}
	}
}

다음 좌표로의 이동이 벽에 막혔는지 확인

wall은 4차원 배열로 (y, x) -> (ny, nx)의 이동이 벽에 막혀있는지가 저장되어 있음

(y==ny || x==nx) 즉, 상하좌우로의 이동은 한가지의 벽에 막혔는지만 확인하면 됨

하지만, 대각선 이동의 경우 2가지의 벽을 확인해야함(사이트에는 문제 설명이 부족해 많이 해맴)

private static boolean isWall(int y, int x, int ny, int nx, int machineDir) {
	// 직선 확산의 경우
	if (y == ny || x == nx) {
		if (wall[y][x][ny][nx]) return true;
	} 
	else {
		// 온풍기 바람 방향이 좌우
		if(machineDir==1||machineDir==2){
			if(wall[y][x][ny][x] || wall[ny][x][ny][nx]) return true;
		}
		// 온풍기 바람 방향이 상하
		else{
			if(wall[y][x][y][nx] ||wall[y][nx][ny][nx])return true;
		}
	}
	return false;
}

 

이러한 작동을 모든 온풍기에 대해 실행

합산은 plus[][]에 저장하여 한번에 실행 

private static void workMachine() {
	plus = new int[N][M];
	for (int i = 0; i < machines.size(); i++) {
		work(machines.get(i).y, machines.get(i).x, machines.get(i).dir);
	}

	for (int y = 0; y < N; y++) {
		for (int x = 0; x < M; x++) {
			map[y][x] += plus[y][x];
		}
	}
}

 

온도 조절

현재 위치의 인접한 칸에 대해 온도 조절

상하좌우 인접한 칸 중 현재 위치보다 온도가 작은 칸에 대해 조절

온도 이동량은 plus[][]에 누적 저장

    - 현재 위치 칸에 대해서는 - (map[y][x]-map[ny][nx])/4

    - 온도가 작은 인접 칸에 대해서는 +(map[y][x]-map[ny][nx])/4

private static void _adjust(int i, int j) {
	for (int d = 1; d < 5; d++) {
		int ny = i + dy[d];
		int nx = j + dx[d];
		if (!isRange(ny, nx)) continue;
		if (wall[i][j][ny][nx]) continue;
        
		if (map[i][j] > map[ny][nx]) {
			int temp = (int) ((map[i][j] - map[ny][nx]) / 4);
			plus[i][j] -= temp;
			plus[ny][nx] += temp;
		}
	}
}

모든 칸에 대한 조사, 온도조절

위의 칸마다의 온도조절을 모든 칸에 적용

온도 이동량이 저장된 plus[][]를 모든 좌표에 계산

또한, 온도가 0이상인 가장자리 칸은 -1 

private static void adjust() {
	plus = new int[N][M];
	for (int i = 0; i < N; i++) {
		for (int j = 0; j < M; j++) {
			if (map[i][j] == 0) continue;
			_adjust(i, j);
		}
	}

	for (int i = 0; i < N; i++) {
		for (int j = 0; j < M; j++) {
			map[i][j] += plus[i][j];
			if (i == 0 || j == 0 || i == N - 1 || j == M - 1) {
				if (map[i][j] > 0) map[i][j] -= 1;
			}
		}
	}
}

 

시뮬레이션

이러한 과정을 (초콜릿 개수 <= 100 && !checkTemp())동안 반복

int choco = 0;
while (choco <= 100 && !checkTemp()) {
	workMachine();
	adjust();
	choco++;
}
System.out.println(choco);

checkTemp()를 이용해 조사해야하는 칸의 온도를 확인

    모든 칸이 K이상이면 정상 종료(return true)

private static boolean checkTemp() {
	for (int i = 0; i < checkList.size(); i++) {
		if (map[checkList.get(i).y][checkList.get(i).x] < K) return false;
	}
	return true;
}

 

전체 코드

 

import java.io.*;
import java.util.*;

public class Main {
    static int N, M, K;
    static int[][] map;
    static int[][] plus;
    static boolean[][][][] wall;
    // 온풍기 정보
    static List<Pos> machines = new ArrayList<>();
    // 온도를 조사해야하는 좌표 리스트
    static List<Pos> checkList = new ArrayList<>();
    // 온풍기 방향
    static int[] dy = {0, 0, 0, -1, 1};
    static int[] dx = {0, 1, -1, 0, 0};
    //온풍기의 방향에 따른 확산 방향
    static int[][] spreadY = {{0, 0, 0}, {0, -1, 1}, {0, -1, 1}, {-1, -1, -1}, {1, 1, 1}};
    static int[][] spreadX = {{0, 0, 0}, {1, 1, 1}, {-1, -1, -1}, {0, -1, 1}, {0, -1, 1}};

    static class Pos {
        int y, x, dir;

        Pos(int y, int x) {
            this.y = y;
            this.x = x;
            this.dir = -1;
        }

        Pos(int y, int x, int dir) {
            this.y = y;
            this.x = x;
            this.dir = dir;
        }
    }

    public static void main(String[] args) throws IOException {
//        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        BufferedReader br = new BufferedReader(new FileReader("a.txt"));

        StringTokenizer st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken());
        M = Integer.parseInt(st.nextToken());
        map = new int[N][M];
        wall = new boolean[N][M][N][M];
        K = Integer.parseInt(st.nextToken());

        for (int i = 0; i < N; i++) {
            st = new StringTokenizer(br.readLine());
            for (int j = 0; j < M; j++) {
                int temp = Integer.parseInt(st.nextToken());
                if (temp == 5) {
                    checkList.add(new Pos(i, j));
                } else if (temp != 0) {
                    machines.add(new Pos(i, j, temp));
                }
            }
        }

        st = new StringTokenizer(br.readLine());
        int cnt = Integer.parseInt(st.nextToken());
        for (int i = 0; i < cnt; i++) {
            st = new StringTokenizer(br.readLine());
            int y = Integer.parseInt(st.nextToken());
            int x = Integer.parseInt(st.nextToken());
            y--;
            x--;
            int t = Integer.parseInt(st.nextToken());
            if (t == 0) {
                wall[y][x][y - 1][x] = true;
                wall[y - 1][x][y][x] = true;
            } else {
                wall[y][x][y][x + 1] = true;
                wall[y][x + 1][y][x] = true;
            }
        }
        int choco = 0;
        while (choco <= 100 && !checkTemp()) {
            workMachine();
            adjust();
            choco++;
        }
        System.out.println(choco);
    }

    // 모든 온풍기작동, 온도 상승을 한번에 합산
    private static void workMachine() {
        plus = new int[N][M];
        for (int i = 0; i < machines.size(); i++) {
            work(machines.get(i).y, machines.get(i).x, machines.get(i).dir);
        }

        for (int y = 0; y < N; y++) {
            for (int x = 0; x < M; x++) {
                map[y][x] += plus[y][x];
            }
        }
    }

    // 하나의 온풍기 작동
    private static void work(int i, int j, int dir) {
        boolean[][] visit = new boolean[N][M];
        Queue<Pos> q = new LinkedList<>();

        int t = 5;
        int ny = i + dy[dir];
        int nx = j + dx[dir];

        visit[ny][nx] = true;
        plus[ny][nx] += 5;
        q.add(new Pos(ny, nx, 2));

        while (!q.isEmpty()) {
            Pos now = q.poll();
            int y = now.y;
            int x = now.x;
            int dis = now.dir;

            if (dis > 5) continue;

            for (int d = 0; d < 3; d++) {
                ny = y + spreadY[dir][d];
                nx = x + spreadX[dir][d];
                if (!isRange(ny, nx)) continue;
                if (visit[ny][nx]) continue;
                if (isWall(y, x, ny, nx, dir)) continue;
                visit[ny][nx] = true;
                plus[ny][nx] += t - dis + 1;
                q.add(new Pos(ny, nx, dis + 1));
            }
        }
    }

    // 현재 위치 -> 다음 위치의 이동에 대해 벽에 막혔는지 확인
    private static boolean isWall(int y, int x, int ny, int nx, int machineDir) {
        if (y == ny || x == nx) {
            if (wall[y][x][ny][nx]) return true;
        } else {
            // 온풍기 바람 방향이 좌우
            if(machineDir==1||machineDir==2){
                if(wall[y][x][ny][x] || wall[ny][x][ny][nx]) return true;
            }
            // 온풍기 바람 방향이 상하
            else{
                if(wall[y][x][y][nx] ||wall[y][nx][ny][nx])return true;
            }
        }
        return false;
    }
    
    // 모든 칸에 대해 온도 조절
    private static void adjust() {
        plus = new int[N][M];
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < M; j++) {
                if (map[i][j] == 0) continue;
                _adjust(i, j);
            }
        }

        for (int i = 0; i < N; i++) {
            for (int j = 0; j < M; j++) {
                map[i][j] += plus[i][j];
                if (i == 0 || j == 0 || i == N - 1 || j == M - 1) {
                    if (map[i][j] > 0) map[i][j] -= 1;
                }
            }
        }
    }
    
    // 현재 좌표(i, j)에 대해 인접 칸의 온도 조절
    private static void _adjust(int i, int j) {
        for (int d = 1; d < 5; d++) {
            int ny = i + dy[d];
            int nx = j + dx[d];
            if (!isRange(ny, nx)) continue;
            if (wall[i][j][ny][nx]) continue;
            if (map[i][j] > map[ny][nx]) {
                int temp = (int) ((map[i][j] - map[ny][nx]) / 4);
                plus[i][j] -= temp;
                plus[ny][nx] += temp;
            }
        }
    }
    
    // 조사해야하는 모든 칸이 K이상인지 확인
    private static boolean checkTemp() {
        for (int i = 0; i < checkList.size(); i++) {
            if (map[checkList.get(i).y][checkList.get(i).x] < K) return false;
        }
        return true;
    }
    
    // 격자 범위를 벗어나는지 확인
    private static boolean isRange(int i, int j) {
        if (i < 0 || j < 0 || i >= N || j >= M) return false;
        else return true;
    }
}