본문 바로가기
백준

백준 2206번 : 벽 부수고 이동하기

by Huiyeong 2021. 4. 9.

 문제

www.acmicpc.net/problem/2206

 

2206번: 벽 부수고 이동하기

N×M의 행렬로 표현되는 맵이 있다. 맵에서 0은 이동할 수 있는 곳을 나타내고, 1은 이동할 수 없는 벽이 있는 곳을 나타낸다. 당신은 (1, 1)에서 (N, M)의 위치까지 이동하려 하는데, 이때 최단 경로

www.acmicpc.net

2206 벽 부수고 이동하기

 

 구현 방법

 벽을 한 개까지 부수고 이동할 수 있기 때문에 방문을 관리하는 visited 배열을 3차원으로 선언하여 좌표와 벽을 부술 수 있는지에 대한 여부를 관리해줍니다. 상하좌우를 검사하여 빈 칸이 일 때에는 그냥 지나가고 벽일 때 현재 좌표에서 벽을 부술 수 있다면 벽을 부수고 지나가줍니다. 이동을 하다가 (N, M) 좌표까지 가게 된다면 반복문을 끝내주고 거리를 리턴, 반복문이 그대로 끝난다면 불가능하므로 -1을 출력해줍니다.

 

 구현 코드

package BOJ.Gold;

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 BOJ2206_벽부수고이동하기 {
	static int N, M, cnt;
	static int[][] map, deltas= {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};
	static Queue<Move> queue;
	static boolean[][][] visited;
	static class Move {
		int r, c, wallB;

		public Move(int r, int c, int wallB) {
			super();
			this.r = r;
			this.c = c;
			this.wallB = wallB;
		}
	}
	
	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());
		map = new int[N][M];
		for (int r=0;r<N;r++) {
			String input = br.readLine();
			for (int c=0;c<M;c++) {
				map[r][c] = input.charAt(c)-'0';
			}
		}
		// 입력완료
		visited = new boolean[N][M][2]; // 관리할 요소 : 좌표, 벽 부수기 가능 여부
		queue = new LinkedList<>();
		queue.offer(new Move(0, 0, 1));
		visited[0][0][1] = true;
		cnt = 1;
		boolean flag = bfs();
		if (flag) System.out.println(cnt);
		else System.out.println(-1);
	}
	
	static boolean bfs() {
		while(!queue.isEmpty()) {
			int size = queue.size();
			for (int i=0;i<size;i++) {
				Move now = queue.poll();
				if (now.r==N-1 && now.c==M-1) return true;
				for (int d=0;d<4;d++) {
					int nr = now.r+deltas[d][0];
					int nc = now.c+deltas[d][1];
					if (isIn(nr, nc)) {
						if (map[nr][nc]==0 && !visited[nr][nc][now.wallB]) {
							visited[nr][nc][now.wallB] = true;
							queue.offer(new Move(nr, nc, now.wallB));
						} else if (map[nr][nc]==1 && now.wallB!=0 && !visited[nr][nc][now.wallB]) { // 벽을 부술 수 있다면 부수기
							visited[nr][nc][now.wallB] = true;
							queue.offer(new Move(nr, nc, 0));
						}
					}
				}
			}
			cnt++;
		}
		
		return false;
	}
	
	static boolean isIn(int nr, int nc) {
		return nr>=0 && nr<N && nc>=0 && nc<M;
	}
}

 

정답!

'백준' 카테고리의 다른 글

백준 9935번 : 문자열 폭발  (0) 2021.07.05
백준 14442번 : 벽 부수고 이동하기 2  (0) 2021.04.10
백준 16953번 : A -> B  (0) 2021.04.09
백준 1755번 : 숫자놀이  (0) 2021.04.09
백준 16235번 : 나무 재테크  (0) 2021.04.09