반응형
01-27 05:03
Today
Total
«   2025/01   »
1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31
관리 메뉴

개발하는 고라니

[백준] 16933번 : 벽 부수고 이동하기 3 본문

Programming/백준

[백준] 16933번 : 벽 부수고 이동하기 3

조용한고라니 2021. 4. 8. 13:13
반응형
 

16933번: 벽 부수고 이동하기 3

첫째 줄에 N(1 ≤ N ≤ 1,000), M(1 ≤ M ≤ 1,000), K(1 ≤ K ≤ 10)이 주어진다. 다음 N개의 줄에 M개의 숫자로 맵이 주어진다. (1, 1)과 (N, M)은 항상 0이라고 가정하자.

www.acmicpc.net


[BFS]

백준의 '벽 부수고 이동하기' 시리즈의 3번째 문제이다. 벽을 부술 수 있는 개수가 K개로 한정되어있고, 벽을 부수는 조건은 '낮'에만 부술 수 있으므로 이 점을 잘 고려해야한다.

 

방문한 것을 체크하는 visit는 4차원 배열로 선언했다. 해당 좌표에 밤에 갔을 때, 낮에 갔을 때, 그리고 벽을 n개 부쉈을 때.

visit[1001][1001][11][2]

 

Queue에 들어가는 원소는 5개이다. 좌표값 y, x 몇 번째 이동했는지? 벽을 몇 개 부쉈는지? 낮인지 밤인지?

 

day : true면 낮, false면 밤

BFS 내부의 d, nd : 0이면 낮, 1이면 밤

# Code </>

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.LinkedList;
import java.util.Queue;
import java.util.StringTokenizer;

public class Main {

    static class Point{
        int y, x, m, w;
        boolean day;
        public Point(int yy, int xx, int move, int wall, boolean day){
            y=yy; x=xx; w=wall; m=move;
            this.day = day;
        }
    }
    static int n, m, k, result = 1000000000;
    static int[] Y = {-1, 1, 0, 0}, X = {0, 0, -1, 1};
    static char[][] map = new char[1001][1001];
    static boolean[][][][] visit = new boolean[1001][1001][11][2];

    static void BFS(){
        Queue<Point> Q = new LinkedList<>();

        visit[1][1][0][0] = true;
        Q.add(new Point(1, 1, 1, 0, true));

        while(!Q.isEmpty()){
            Point p = Q.poll();

            int y = p.y;
            int x = p.x;
            int w = p.w;
            int d = p.day? 0:1;
            int nd = d == 0? 1:0;

            int move = p.m;
            if(y == n && x == m) {
                result = Math.min(result, move);
            }

            //제자리 머물기
            if(!visit[y][x][w][nd]){
                visit[y][x][w][nd] = true;
                Q.add(new Point(y, x, move + 1, w, !p.day));
            }
            for(int a=0; a<4; a++){
                int ny = y+Y[a];
                int nx = x+X[a];

                if(ny < 1 || nx < 1 || ny > n || nx > m || visit[ny][nx][w][nd]) continue;

                if(map[ny][nx] == '0') {
                    visit[ny][nx][w][nd] = true;
                    Q.add(new Point(ny, nx, move + 1, w, !p.day));
                }

                else if(map[ny][nx] == '1' && p.day && w < k) {
                    visit[ny][nx][w+1][nd] = true;
                    Q.add(new Point(ny, nx, move + 1, w + 1, false));
                }
            }

        }
    }

    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st = new StringTokenizer(br.readLine());

        n = Integer.parseInt(st.nextToken());
        m = Integer.parseInt(st.nextToken());
        k = Integer.parseInt(st.nextToken());

        for(int i=1; i<=n; i++){
            char[] str = br.readLine().toCharArray();
            for(int j=1; j<=m; j++)
                map[i][j] = str[j-1];
        }

        BFS();
        System.out.println(result == 1000000000? -1:result);
    }
}
반응형
Comments